mirror of https://github.com/apache/activemq.git
git-svn-id: https://svn.apache.org/repos/asf/incubator/activemq/branches/activemq-4.0@443423 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
parent
beb72f6403
commit
dc789d17e6
|
@ -61,6 +61,7 @@ import org.apache.activemq.state.CommandVisitor;
|
||||||
import org.apache.activemq.state.ConsumerState;
|
import org.apache.activemq.state.ConsumerState;
|
||||||
import org.apache.activemq.state.ProducerState;
|
import org.apache.activemq.state.ProducerState;
|
||||||
import org.apache.activemq.state.SessionState;
|
import org.apache.activemq.state.SessionState;
|
||||||
|
import org.apache.activemq.state.TransactionState;
|
||||||
import org.apache.activemq.thread.Task;
|
import org.apache.activemq.thread.Task;
|
||||||
import org.apache.activemq.thread.TaskRunner;
|
import org.apache.activemq.thread.TaskRunner;
|
||||||
import org.apache.activemq.thread.TaskRunnerFactory;
|
import org.apache.activemq.thread.TaskRunnerFactory;
|
||||||
|
@ -307,7 +308,12 @@ public abstract class AbstractConnection implements Service, Connection, Task, C
|
||||||
if( cs!=null ) {
|
if( cs!=null ) {
|
||||||
context = cs.getContext();
|
context = cs.getContext();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Avoid replaying dup commands
|
||||||
|
if( cs.getTransactionState(info.getTransactionId())==null ) {
|
||||||
|
cs.addTransactionState(info.getTransactionId());
|
||||||
broker.beginTransaction(context, info.getTransactionId());
|
broker.beginTransaction(context, info.getTransactionId());
|
||||||
|
}
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -324,9 +330,22 @@ public abstract class AbstractConnection implements Service, Connection, Task, C
|
||||||
if( cs!=null ) {
|
if( cs!=null ) {
|
||||||
context = cs.getContext();
|
context = cs.getContext();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
TransactionState transactionState = cs.getTransactionState(info.getTransactionId());
|
||||||
|
if( transactionState == null )
|
||||||
|
throw new IllegalStateException("Cannot prepare a transaction that had not been started: "+info.getTransactionId());
|
||||||
|
|
||||||
|
// Avoid dups.
|
||||||
|
if( !transactionState.isPrepared() ) {
|
||||||
|
transactionState.setPrepared(true);
|
||||||
int result = broker.prepareTransaction(context, info.getTransactionId());
|
int result = broker.prepareTransaction(context, info.getTransactionId());
|
||||||
|
transactionState.setPreparedResult(result);
|
||||||
IntegerResponse response = new IntegerResponse(result);
|
IntegerResponse response = new IntegerResponse(result);
|
||||||
return response;
|
return response;
|
||||||
|
} else {
|
||||||
|
IntegerResponse response = new IntegerResponse(transactionState.getPreparedResult());
|
||||||
|
return response;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public Response processCommitTransactionOnePhase(TransactionInfo info) throws Exception {
|
public Response processCommitTransactionOnePhase(TransactionInfo info) throws Exception {
|
||||||
|
@ -335,8 +354,12 @@ public abstract class AbstractConnection implements Service, Connection, Task, C
|
||||||
if( cs!=null ) {
|
if( cs!=null ) {
|
||||||
context = cs.getContext();
|
context = cs.getContext();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
cs.removeTransactionState(info.getTransactionId());
|
||||||
broker.commitTransaction(context, info.getTransactionId(), true);
|
broker.commitTransaction(context, info.getTransactionId(), true);
|
||||||
|
|
||||||
return null;
|
return null;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public Response processCommitTransactionTwoPhase(TransactionInfo info) throws Exception {
|
public Response processCommitTransactionTwoPhase(TransactionInfo info) throws Exception {
|
||||||
|
@ -345,6 +368,8 @@ public abstract class AbstractConnection implements Service, Connection, Task, C
|
||||||
if( cs!=null ) {
|
if( cs!=null ) {
|
||||||
context = cs.getContext();
|
context = cs.getContext();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
cs.removeTransactionState(info.getTransactionId());
|
||||||
broker.commitTransaction(context, info.getTransactionId(), false);
|
broker.commitTransaction(context, info.getTransactionId(), false);
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
@ -355,6 +380,8 @@ public abstract class AbstractConnection implements Service, Connection, Task, C
|
||||||
if( cs!=null ) {
|
if( cs!=null ) {
|
||||||
context = cs.getContext();
|
context = cs.getContext();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
cs.removeTransactionState(info.getTransactionId());
|
||||||
broker.rollbackTransaction(context, info.getTransactionId());
|
broker.rollbackTransaction(context, info.getTransactionId());
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
@ -381,10 +408,32 @@ public abstract class AbstractConnection implements Service, Connection, Task, C
|
||||||
|
|
||||||
|
|
||||||
public Response processMessage(Message messageSend) throws Exception {
|
public Response processMessage(Message messageSend) throws Exception {
|
||||||
|
|
||||||
ProducerId producerId = messageSend.getProducerId();
|
ProducerId producerId = messageSend.getProducerId();
|
||||||
ConnectionState state = lookupConnectionState(producerId);
|
ConnectionState state = lookupConnectionState(producerId);
|
||||||
ConnectionContext context = state.getContext();
|
ConnectionContext context = state.getContext();
|
||||||
|
|
||||||
|
// If the message originates from this client connection,
|
||||||
|
// then, finde the associated producer state so we can do some dup detection.
|
||||||
|
ProducerState producerState=null;
|
||||||
|
if( messageSend.getMessageId().getProducerId().equals( messageSend.getProducerId() ) ) {
|
||||||
|
SessionState ss = state.getSessionState(producerId.getParentId());
|
||||||
|
if( ss == null )
|
||||||
|
throw new IllegalStateException("Cannot send from a session that had not been registered: "+producerId.getParentId());
|
||||||
|
producerState = ss.getProducerState(producerId);
|
||||||
|
}
|
||||||
|
|
||||||
|
if( producerState == null ) {
|
||||||
broker.send(context, messageSend);
|
broker.send(context, messageSend);
|
||||||
|
} else {
|
||||||
|
// Avoid Dups.
|
||||||
|
long seq = messageSend.getMessageId().getProducerSequenceId();
|
||||||
|
if( seq > producerState.getLastSequenceId() ) {
|
||||||
|
producerState.setLastSequenceId(seq);
|
||||||
|
broker.send(context, messageSend);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -30,6 +30,7 @@ import org.apache.activemq.command.ConnectionInfo;
|
||||||
import org.apache.activemq.command.DestinationInfo;
|
import org.apache.activemq.command.DestinationInfo;
|
||||||
import org.apache.activemq.command.SessionId;
|
import org.apache.activemq.command.SessionId;
|
||||||
import org.apache.activemq.command.SessionInfo;
|
import org.apache.activemq.command.SessionInfo;
|
||||||
|
import org.apache.activemq.command.TransactionId;
|
||||||
|
|
||||||
import edu.emory.mathcs.backport.java.util.concurrent.ConcurrentHashMap;
|
import edu.emory.mathcs.backport.java.util.concurrent.ConcurrentHashMap;
|
||||||
import edu.emory.mathcs.backport.java.util.concurrent.atomic.AtomicBoolean;
|
import edu.emory.mathcs.backport.java.util.concurrent.atomic.AtomicBoolean;
|
||||||
|
@ -37,6 +38,7 @@ import edu.emory.mathcs.backport.java.util.concurrent.atomic.AtomicBoolean;
|
||||||
public class ConnectionState {
|
public class ConnectionState {
|
||||||
|
|
||||||
final ConnectionInfo info;
|
final ConnectionInfo info;
|
||||||
|
private final ConcurrentHashMap transactions = new ConcurrentHashMap();
|
||||||
private final ConcurrentHashMap sessions = new ConcurrentHashMap();
|
private final ConcurrentHashMap sessions = new ConcurrentHashMap();
|
||||||
private final List tempDestinations = Collections.synchronizedList(new ArrayList());
|
private final List tempDestinations = Collections.synchronizedList(new ArrayList());
|
||||||
private final AtomicBoolean shutdown = new AtomicBoolean(false);
|
private final AtomicBoolean shutdown = new AtomicBoolean(false);
|
||||||
|
@ -65,6 +67,20 @@ public class ConnectionState {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void addTransactionState(TransactionId id) {
|
||||||
|
checkShutdown();
|
||||||
|
transactions.put(id, new TransactionState(id));
|
||||||
|
}
|
||||||
|
public TransactionState getTransactionState(TransactionId id) {
|
||||||
|
return (TransactionState)transactions.get(id);
|
||||||
|
}
|
||||||
|
public Collection getTransactionStates() {
|
||||||
|
return transactions.values();
|
||||||
|
}
|
||||||
|
public TransactionState removeTransactionState(TransactionId id) {
|
||||||
|
return (TransactionState) transactions.remove(id);
|
||||||
|
}
|
||||||
|
|
||||||
public void addSession(SessionInfo info) {
|
public void addSession(SessionInfo info) {
|
||||||
checkShutdown();
|
checkShutdown();
|
||||||
sessions.put(info.getSessionId(), new SessionState(info));
|
sessions.put(info.getSessionId(), new SessionState(info));
|
||||||
|
|
|
@ -54,21 +54,39 @@ import edu.emory.mathcs.backport.java.util.concurrent.ConcurrentHashMap;
|
||||||
*/
|
*/
|
||||||
public class ConnectionStateTracker implements CommandVisitor {
|
public class ConnectionStateTracker implements CommandVisitor {
|
||||||
|
|
||||||
private final static Response TRACKED_RESPONSE_MARKER = new Response();
|
private final static Tracked TRACKED_RESPONSE_MARKER = new Tracked(null);
|
||||||
|
|
||||||
boolean trackTransactions = false;
|
private boolean trackTransactions = false;
|
||||||
boolean trackMessages = false;
|
|
||||||
boolean trackAcks = false;
|
|
||||||
|
|
||||||
private boolean restoreSessions=true;
|
private boolean restoreSessions=true;
|
||||||
boolean restoreConsumers=true;
|
private boolean restoreConsumers=true;
|
||||||
private boolean restoreProducers=true;
|
private boolean restoreProducers=true;
|
||||||
|
private boolean restoreTransaction=true;
|
||||||
|
|
||||||
protected final ConcurrentHashMap connectionStates = new ConcurrentHashMap();
|
protected final ConcurrentHashMap connectionStates = new ConcurrentHashMap();
|
||||||
|
|
||||||
public boolean track(Command command) throws IOException {
|
private class RemoveTransactionAction implements Runnable {
|
||||||
|
private final TransactionInfo info;
|
||||||
|
public RemoveTransactionAction(TransactionInfo info) {
|
||||||
|
this.info = info;
|
||||||
|
}
|
||||||
|
public void run() {
|
||||||
|
ConnectionId connectionId = info.getConnectionId();
|
||||||
|
ConnectionState cs = (ConnectionState) connectionStates.get(connectionId);
|
||||||
|
cs.removeTransactionState(info.getTransactionId());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
*
|
||||||
|
* @param command
|
||||||
|
* @return null if the command is not state tracked.
|
||||||
|
* @throws IOException
|
||||||
|
*/
|
||||||
|
public Tracked track(Command command) throws IOException {
|
||||||
try {
|
try {
|
||||||
return command.visit(this)!=null;
|
return (Tracked) command.visit(this);
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
throw e;
|
throw e;
|
||||||
} catch (Throwable e) {
|
} catch (Throwable e) {
|
||||||
|
@ -85,6 +103,19 @@ public class ConnectionStateTracker implements CommandVisitor {
|
||||||
|
|
||||||
if( restoreSessions )
|
if( restoreSessions )
|
||||||
restoreSessions(transport, connectionState);
|
restoreSessions(transport, connectionState);
|
||||||
|
|
||||||
|
if( restoreTransaction )
|
||||||
|
restoreTransactions(transport, connectionState);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void restoreTransactions(Transport transport, ConnectionState connectionState) throws IOException {
|
||||||
|
for (Iterator iter = connectionState.getTransactionStates().iterator(); iter.hasNext();) {
|
||||||
|
TransactionState transactionState = (TransactionState) iter.next();
|
||||||
|
for (Iterator iterator = transactionState.getCommands().iterator(); iterator.hasNext();) {
|
||||||
|
Command command = (Command) iterator.next();
|
||||||
|
transport.oneway(command);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -226,26 +257,103 @@ public class ConnectionStateTracker implements CommandVisitor {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
public Response processMessage(Message send) throws Exception {
|
public Response processMessage(Message send) throws Exception {
|
||||||
|
if( trackTransactions && send.getTransactionId() != null ) {
|
||||||
|
ConnectionId connectionId = send.getProducerId().getParentId().getParentId();
|
||||||
|
ConnectionState cs = (ConnectionState) connectionStates.get(connectionId);
|
||||||
|
TransactionState transactionState = cs.getTransactionState(send.getTransactionId());
|
||||||
|
transactionState.addCommand(send);
|
||||||
|
return TRACKED_RESPONSE_MARKER;
|
||||||
|
}
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
public Response processMessageAck(MessageAck ack) throws Exception {
|
public Response processMessageAck(MessageAck ack) throws Exception {
|
||||||
|
if( trackTransactions && ack.getTransactionId() != null ) {
|
||||||
|
ConnectionId connectionId = ack.getConsumerId().getParentId().getParentId();
|
||||||
|
ConnectionState cs = (ConnectionState) connectionStates.get(connectionId);
|
||||||
|
TransactionState transactionState = cs.getTransactionState(ack.getTransactionId());
|
||||||
|
transactionState.addCommand(ack);
|
||||||
|
return TRACKED_RESPONSE_MARKER;
|
||||||
|
}
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
public Response processBeginTransaction(TransactionInfo info) throws Exception {
|
public Response processBeginTransaction(TransactionInfo info) throws Exception {
|
||||||
|
if( trackTransactions ) {
|
||||||
|
ConnectionId connectionId = info.getConnectionId();
|
||||||
|
ConnectionState cs = (ConnectionState) connectionStates.get(connectionId);
|
||||||
|
cs.addTransactionState(info.getTransactionId());
|
||||||
|
return TRACKED_RESPONSE_MARKER;
|
||||||
|
}
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
public Response processPrepareTransaction(TransactionInfo info) throws Exception {
|
public Response processPrepareTransaction(TransactionInfo info) throws Exception {
|
||||||
|
if( trackTransactions ) {
|
||||||
|
ConnectionId connectionId = info.getConnectionId();
|
||||||
|
ConnectionState cs = (ConnectionState) connectionStates.get(connectionId);
|
||||||
|
TransactionState transactionState = cs.getTransactionState(info.getTransactionId());
|
||||||
|
transactionState.addCommand(info);
|
||||||
|
return TRACKED_RESPONSE_MARKER;
|
||||||
|
}
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
public Response processCommitTransactionOnePhase(TransactionInfo info) throws Exception {
|
public Response processCommitTransactionOnePhase(TransactionInfo info) throws Exception {
|
||||||
|
if( trackTransactions ) {
|
||||||
|
ConnectionId connectionId = info.getConnectionId();
|
||||||
|
ConnectionState cs = (ConnectionState) connectionStates.get(connectionId);
|
||||||
|
TransactionState transactionState = cs.getTransactionState(info.getTransactionId());
|
||||||
|
if( transactionState !=null ) {
|
||||||
|
transactionState.addCommand(info);
|
||||||
|
return new Tracked(new RemoveTransactionAction(info));
|
||||||
|
}
|
||||||
|
}
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
public Response processCommitTransactionTwoPhase(TransactionInfo info) throws Exception {
|
public Response processCommitTransactionTwoPhase(TransactionInfo info) throws Exception {
|
||||||
|
if( trackTransactions ) {
|
||||||
|
ConnectionId connectionId = info.getConnectionId();
|
||||||
|
ConnectionState cs = (ConnectionState) connectionStates.get(connectionId);
|
||||||
|
TransactionState transactionState = cs.getTransactionState(info.getTransactionId());
|
||||||
|
if( transactionState !=null ) {
|
||||||
|
transactionState.addCommand(info);
|
||||||
|
return new Tracked(new RemoveTransactionAction(info));
|
||||||
|
}
|
||||||
|
}
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
public Response processRollbackTransaction(TransactionInfo info) throws Exception {
|
public Response processRollbackTransaction(TransactionInfo info) throws Exception {
|
||||||
|
if( trackTransactions ) {
|
||||||
|
ConnectionId connectionId = info.getConnectionId();
|
||||||
|
ConnectionState cs = (ConnectionState) connectionStates.get(connectionId);
|
||||||
|
TransactionState transactionState = cs.getTransactionState(info.getTransactionId());
|
||||||
|
if( transactionState !=null ) {
|
||||||
|
transactionState.addCommand(info);
|
||||||
|
return new Tracked(new RemoveTransactionAction(info));
|
||||||
|
}
|
||||||
|
}
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public Response processEndTransaction(TransactionInfo info) throws Exception {
|
||||||
|
if( trackTransactions ) {
|
||||||
|
ConnectionId connectionId = info.getConnectionId();
|
||||||
|
ConnectionState cs = (ConnectionState) connectionStates.get(connectionId);
|
||||||
|
TransactionState transactionState = cs.getTransactionState(info.getTransactionId());
|
||||||
|
transactionState.addCommand(info);
|
||||||
|
return TRACKED_RESPONSE_MARKER;
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Response processRecoverTransactions(TransactionInfo info) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
public Response processForgetTransaction(TransactionInfo info) throws Exception {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
public Response processWireFormat(WireFormatInfo info) throws Exception {
|
public Response processWireFormat(WireFormatInfo info) throws Exception {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
@ -259,18 +367,6 @@ public class ConnectionStateTracker implements CommandVisitor {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
public Response processRecoverTransactions(TransactionInfo info) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
public Response processForgetTransaction(TransactionInfo info) throws Exception {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
public Response processEndTransaction(TransactionInfo info) throws Exception {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
public Response processFlush(FlushCommand command) throws Exception {
|
public Response processFlush(FlushCommand command) throws Exception {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
@ -302,4 +398,20 @@ public class ConnectionStateTracker implements CommandVisitor {
|
||||||
public void setRestoreSessions(boolean restoreSessions) {
|
public void setRestoreSessions(boolean restoreSessions) {
|
||||||
this.restoreSessions = restoreSessions;
|
this.restoreSessions = restoreSessions;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public boolean isTrackTransactions() {
|
||||||
|
return trackTransactions;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setTrackTransactions(boolean trackTransactions) {
|
||||||
|
this.trackTransactions = trackTransactions;
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean isRestoreTransaction() {
|
||||||
|
return restoreTransaction;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setRestoreTransaction(boolean restoreTransaction) {
|
||||||
|
this.restoreTransaction = restoreTransaction;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -22,6 +22,7 @@ import org.apache.activemq.command.ProducerInfo;
|
||||||
|
|
||||||
public class ProducerState {
|
public class ProducerState {
|
||||||
final ProducerInfo info;
|
final ProducerInfo info;
|
||||||
|
private long lastSequenceId=-1;
|
||||||
|
|
||||||
public ProducerState(ProducerInfo info) {
|
public ProducerState(ProducerInfo info) {
|
||||||
this.info = info;
|
this.info = info;
|
||||||
|
@ -32,4 +33,10 @@ public class ProducerState {
|
||||||
public ProducerInfo getInfo() {
|
public ProducerInfo getInfo() {
|
||||||
return info;
|
return info;
|
||||||
}
|
}
|
||||||
|
public void setLastSequenceId(long lastSequenceId) {
|
||||||
|
this.lastSequenceId = lastSequenceId;
|
||||||
|
}
|
||||||
|
public long getLastSequenceId() {
|
||||||
|
return lastSequenceId;
|
||||||
|
}
|
||||||
}
|
}
|
|
@ -70,10 +70,12 @@ public class SessionState {
|
||||||
public Set getProducerIds() {
|
public Set getProducerIds() {
|
||||||
return producers.keySet();
|
return producers.keySet();
|
||||||
}
|
}
|
||||||
|
|
||||||
public Collection getProducerStates() {
|
public Collection getProducerStates() {
|
||||||
return producers.values();
|
return producers.values();
|
||||||
}
|
}
|
||||||
|
public ProducerState getProducerState(ProducerId producerId) {
|
||||||
|
return (ProducerState) producers.get(producerId);
|
||||||
|
}
|
||||||
|
|
||||||
public Collection getConsumerStates() {
|
public Collection getConsumerStates() {
|
||||||
return consumers.values();
|
return consumers.values();
|
||||||
|
|
|
@ -0,0 +1,27 @@
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
package org.apache.activemq.state;
|
||||||
|
|
||||||
|
import org.apache.activemq.command.Response;
|
||||||
|
|
||||||
|
public class Tracked extends Response {
|
||||||
|
|
||||||
|
private Runnable runnable;
|
||||||
|
|
||||||
|
public Tracked(Runnable runnable) {
|
||||||
|
this.runnable = runnable;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void onResponses() {
|
||||||
|
if( runnable != null ) {
|
||||||
|
runnable.run();
|
||||||
|
runnable=null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean isWaitingForResponse() {
|
||||||
|
return runnable!=null;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,79 @@
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* Copyright 2005-2006 The Apache Software Foundation
|
||||||
|
*
|
||||||
|
* 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 org.apache.activemq.state;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
import org.apache.activemq.command.Command;
|
||||||
|
import org.apache.activemq.command.TransactionId;
|
||||||
|
|
||||||
|
import edu.emory.mathcs.backport.java.util.concurrent.atomic.AtomicBoolean;
|
||||||
|
|
||||||
|
public class TransactionState {
|
||||||
|
final TransactionId id;
|
||||||
|
|
||||||
|
public final ArrayList commands = new ArrayList();
|
||||||
|
private final AtomicBoolean shutdown = new AtomicBoolean(false);
|
||||||
|
|
||||||
|
private boolean prepared;
|
||||||
|
|
||||||
|
private int preparedResult;
|
||||||
|
|
||||||
|
public TransactionState(TransactionId id) {
|
||||||
|
this.id = id;
|
||||||
|
}
|
||||||
|
public String toString() {
|
||||||
|
return id.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void addCommand(Command operation) {
|
||||||
|
checkShutdown();
|
||||||
|
commands.add(operation);
|
||||||
|
}
|
||||||
|
|
||||||
|
public List getCommands() {
|
||||||
|
return commands;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void checkShutdown() {
|
||||||
|
if( shutdown.get() )
|
||||||
|
throw new IllegalStateException("Disposed");
|
||||||
|
}
|
||||||
|
|
||||||
|
public void shutdown() {
|
||||||
|
shutdown.set(false);
|
||||||
|
}
|
||||||
|
public TransactionId getId() {
|
||||||
|
return id;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setPrepared(boolean prepared) {
|
||||||
|
this.prepared = prepared;
|
||||||
|
}
|
||||||
|
public boolean isPrepared() {
|
||||||
|
return prepared;
|
||||||
|
}
|
||||||
|
public void setPreparedResult(int preparedResult) {
|
||||||
|
this.preparedResult = preparedResult;
|
||||||
|
}
|
||||||
|
public int getPreparedResult() {
|
||||||
|
return preparedResult;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -28,6 +28,7 @@ import org.apache.activemq.command.BrokerInfo;
|
||||||
import org.apache.activemq.command.Command;
|
import org.apache.activemq.command.Command;
|
||||||
import org.apache.activemq.command.Response;
|
import org.apache.activemq.command.Response;
|
||||||
import org.apache.activemq.state.ConnectionStateTracker;
|
import org.apache.activemq.state.ConnectionStateTracker;
|
||||||
|
import org.apache.activemq.state.Tracked;
|
||||||
import org.apache.activemq.thread.DefaultThreadPools;
|
import org.apache.activemq.thread.DefaultThreadPools;
|
||||||
import org.apache.activemq.thread.Task;
|
import org.apache.activemq.thread.Task;
|
||||||
import org.apache.activemq.thread.TaskRunner;
|
import org.apache.activemq.thread.TaskRunner;
|
||||||
|
@ -86,7 +87,10 @@ public class FailoverTransport implements CompositeTransport {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (command.isResponse()) {
|
if (command.isResponse()) {
|
||||||
requestMap.remove(new Integer(((Response) command).getCorrelationId()));
|
Object object = requestMap.remove(new Integer(((Response) command).getCorrelationId()));
|
||||||
|
if( object!=null && object.getClass() == Tracked.class ) {
|
||||||
|
((Tracked)object).onResponses();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
if (!initialized){
|
if (!initialized){
|
||||||
if (command.isBrokerInfo()){
|
if (command.isBrokerInfo()){
|
||||||
|
@ -132,6 +136,8 @@ public class FailoverTransport implements CompositeTransport {
|
||||||
|
|
||||||
public FailoverTransport() throws InterruptedIOException {
|
public FailoverTransport() throws InterruptedIOException {
|
||||||
|
|
||||||
|
stateTracker.setTrackTransactions(true);
|
||||||
|
|
||||||
// Setup a task that is used to reconnect the a connection async.
|
// Setup a task that is used to reconnect the a connection async.
|
||||||
reconnectTask = DefaultThreadPools.getDefaultTaskRunnerFactory().createTaskRunner(new Task() {
|
reconnectTask = DefaultThreadPools.getDefaultTaskRunnerFactory().createTaskRunner(new Task() {
|
||||||
|
|
||||||
|
@ -368,7 +374,10 @@ public class FailoverTransport implements CompositeTransport {
|
||||||
// the state tracker,
|
// the state tracker,
|
||||||
// then hold it in the requestMap so that we can replay
|
// then hold it in the requestMap so that we can replay
|
||||||
// it later.
|
// it later.
|
||||||
if (!stateTracker.track(command) && command.isResponseRequired()) {
|
Tracked tracked = stateTracker.track(command);
|
||||||
|
if( tracked!=null && tracked.isWaitingForResponse() ) {
|
||||||
|
requestMap.put(new Integer(command.getCommandId()), tracked);
|
||||||
|
} else if ( tracked==null && command.isResponseRequired()) {
|
||||||
requestMap.put(new Integer(command.getCommandId()), command);
|
requestMap.put(new Integer(command.getCommandId()), command);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -376,15 +385,22 @@ public class FailoverTransport implements CompositeTransport {
|
||||||
try {
|
try {
|
||||||
connectedTransport.oneway(command);
|
connectedTransport.oneway(command);
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
// If there is an IOException in the send, remove the command from the requestMap
|
|
||||||
if (!stateTracker.track(command) && command.isResponseRequired()) {
|
// If the command was not tracked.. we will retry in this method
|
||||||
requestMap.remove(new Integer(command.getCommandId()), command);
|
if( tracked==null ) {
|
||||||
|
|
||||||
|
// since we will retry in this method.. take it out of the request
|
||||||
|
// map so that it is not sent 2 times on recovery
|
||||||
|
if( command.isResponseRequired() ) {
|
||||||
|
requestMap.remove(new Integer(command.getCommandId()));
|
||||||
}
|
}
|
||||||
|
|
||||||
// Rethrow the exception so it will handled by the outer catch
|
// Rethrow the exception so it will handled by the outer catch
|
||||||
throw e;
|
throw e;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
return;
|
return;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -340,7 +340,7 @@ public class FanoutTransport implements CompositeTransport {
|
||||||
// then hold it in the requestMap so that we can replay
|
// then hold it in the requestMap so that we can replay
|
||||||
// it later.
|
// it later.
|
||||||
boolean fanout = isFanoutCommand(command);
|
boolean fanout = isFanoutCommand(command);
|
||||||
if (!stateTracker.track(command) && command.isResponseRequired() ) {
|
if (stateTracker.track(command)==null && command.isResponseRequired() ) {
|
||||||
int size = fanout ? minAckCount : 1;
|
int size = fanout ? minAckCount : 1;
|
||||||
requestMap.put(new Integer(command.getCommandId()), new RequestCounter(command, size));
|
requestMap.put(new Integer(command.getCommandId()), new RequestCounter(command, size));
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue