com.gemstone.gemfire.cache.client.internal.TXSynchronizationOp Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of gemfire-core Show documentation
Show all versions of gemfire-core Show documentation
SnappyData store based off Pivotal GemFireXD
The newest version!
/*
* Copyright (c) 2010-2015 Pivotal Software, Inc. All rights reserved.
*
* 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. See accompanying
* LICENSE file.
*/
package com.gemstone.gemfire.cache.client.internal;
import com.gemstone.gemfire.GemFireException;
import com.gemstone.gemfire.cache.CommitConflictException;
import com.gemstone.gemfire.cache.SynchronizationCommitConflictException;
import com.gemstone.gemfire.cache.client.ServerOperationException;
import com.gemstone.gemfire.i18n.LogWriterI18n;
import com.gemstone.gemfire.internal.cache.TXManagerImpl;
import com.gemstone.gemfire.internal.cache.TXRemoteCommitMessage;
import com.gemstone.gemfire.internal.cache.tier.MessageType;
import com.gemstone.gemfire.internal.cache.tier.sockets.Message;
import com.gemstone.gemfire.internal.cache.tier.sockets.Part;
/**
* TXSynchronizationOp sends JTA beforeCompletion and afterCompletion
* messages to the server pool.
*
* @author bruce
*
*/
public class TXSynchronizationOp {
public static enum CompletionType {
BEFORE_COMPLETION, AFTER_COMPLETION
}
/**
* @param pool
* @param status - the status of an afterCompletion notification
* @param txId - the transaction identifier
* @param type - BEFORE_COMPLETION or AFTER_COMPLETION
* @return the server's commit message
*/
public static TXRemoteCommitMessage execute(InternalPool pool, int status, int txId, CompletionType type) {
Impl impl = new Impl(pool.getLoggerI18n(), status, txId, type);
pool.execute(impl);
return impl.tXCommitMessageResponse;
}
static class Impl extends AbstractOp {
private final int status;
private final CompletionType type;
TXRemoteCommitMessage tXCommitMessageResponse;
/**
* @param logger
* @param status
* @param type
*/
public Impl(LogWriterI18n logger, int status, int txId, CompletionType type) {
super(logger, MessageType.TX_SYNCHRONIZATION,
(type==CompletionType.AFTER_COMPLETION)? 3 : 2);
this.status = status;
this.type = type;
getMessage().addIntPart(type.ordinal());
getMessage().addIntPart(txId);
if (type == CompletionType.AFTER_COMPLETION) {
getMessage().addIntPart(status);
}
}
@Override
public String toString() {
return "TXSynchronization(threadTxId=" + TXManagerImpl.getCurrentTXId()
+"; "+this.type + "; status=" + this.status + ")";
}
@Override
protected void processAck(Message msg, String opName)
throws Exception
{
final int msgType = msg.getMessageType();
if (msgType == MessageType.REPLY) {
return;
} else {
Part part = msg.getPart(0);
if (msgType == MessageType.EXCEPTION) {
Throwable t = (Throwable) part.getObject();
if (t instanceof CommitConflictException ||
t instanceof SynchronizationCommitConflictException) {
throw (GemFireException)t;
}
}
super.processAck(msg, opName);
}
}
/* (non-Javadoc)
* @see com.gemstone.gemfire.cache.client.internal.AbstractOp#processResponse(com.gemstone.gemfire.internal.cache.tier.sockets.Message)
*/
@Override
protected Object processResponse(Message msg) throws Exception {
if (this.type == CompletionType.BEFORE_COMPLETION) {
try {
processAck(msg, type.toString());
} catch (ServerOperationException e) {
if (e.getCause() instanceof SynchronizationCommitConflictException) {
throw (SynchronizationCommitConflictException)e.getCause();
}
}
return null;
} else {
TXRemoteCommitMessage rcs = (TXRemoteCommitMessage)processObjResponse(msg, this.type.toString());
this.tXCommitMessageResponse = rcs;
return rcs;
}
}
/* (non-Javadoc)
* @see com.gemstone.gemfire.cache.client.internal.AbstractOp#isErrorResponse(int)
*/
@Override
protected boolean isErrorResponse(int msgType) {
return msgType == MessageType.REQUESTDATAERROR;
}
@Override
protected long startAttempt(ConnectionStats stats) {
return stats.startTxSynchronization();
}
@Override
protected void endSendAttempt(ConnectionStats stats, long start) {
stats.endTxSynchronizationSend(start, hasFailed());
}
@Override
protected void endAttempt(ConnectionStats stats, long start) {
stats.endTxSynchronization(start, hasTimedOut(), hasFailed());
}
@Override
protected void processSecureBytes(Connection cnx, Message message)
throws Exception {
}
@Override
protected boolean needsUserId() {
return false;
}
@Override
protected void sendMessage(Connection cnx) throws Exception {
getMessage().setEarlyAck((byte)(getMessage().getEarlyAckByte() & Message.MESSAGE_HAS_SECURE_PART));
getMessage().send(false);
}
}
}