/*******************************************************************************
* gMix open source project - https://svs.informatik.uni-hamburg.de/gmix/
* Copyright (C) 2014 SVS
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*******************************************************************************/
package userGeneratedContent.testbedPlugIns.layerPlugIns.layer4transport.orderPreservingForwarder_v0_001;
import java.util.PriorityQueue;
import staticContent.framework.controller.Implementation;
import staticContent.framework.controller.Layer3OutputStrategyMixController;
import staticContent.framework.interfaces.Layer4TransportMix;
import staticContent.framework.message.Reply;
import staticContent.framework.message.Request;
import staticContent.framework.userDatabase.User;
import staticContent.framework.userDatabase.UserAttachment;
import staticContent.framework.util.Util;
public class MixPlugIn extends Implementation implements Layer4TransportMix {
private Layer3OutputStrategyMixController layer3controller;
@Override
public void constructor() {
System.out.println("loaded " +this +" on " +anonNode.PUBLIC_PSEUDONYM);
this.layer3controller = anonNode.getOutputStrategyLayerControllerMix();
}
@Override
public void initialize() {
}
@Override
public void begin() {
}
@Override
public void forwardRequest(Request request) {
UserData userData = request.getOwner().getAttachment(this, UserData.class);
if (userData == null)
userData = new UserData(request.getOwner());
if (userData.nextSequenceNumberReceive == Integer.MAX_VALUE)
displayErrorMessage();
byte[][] splitted = Util.split(4, request.getByteMessage());
request.setByteMessage(splitted[1]);
QueueElement<Request> requestElement = new QueueElement<Request>(Util.byteArrayToInt(splitted[0]), request);
if (requestElement.getSequenceNumber() == userData.nextSequenceNumberReceive) { // we are waiting for this message
//System.out.println("waiting for " +userData.nextSequenceNumberReceive +", received: " +requestElement.getSequenceNumber() +"yay"); // TODO: remove
userData.nextSequenceNumberReceive++;
//System.out.println("forwarding " +requestElement.getSequenceNumber());
anonNode.forwardToLayer5(requestElement.getMessage());
int numberOfQueuedRequests = userData.queuedRequests.size();
for (int i=0; i<numberOfQueuedRequests; i++) {
QueueElement<Request> e = userData.queuedRequests.peek();
if (e.getSequenceNumber() == userData.nextSequenceNumberReceive) { // we are waiting for this message
//System.out.println("waiting for " +userData.nextSequenceNumberReceive +", received: " +e.getSequenceNumber()); // TODO: remove
anonNode.forwardToLayer5(userData.queuedRequests.remove().getMessage());
userData.nextSequenceNumberReceive++;
} else {
break;
}
}
} else { // we are not waiting for this message
//System.out.println("waiting for " +userData.nextSequenceNumberReceive +", received: " +requestElement.getSequenceNumber()); // TODO: remove
userData.queuedRequests.add(requestElement);
}
}
private void displayErrorMessage() {
throw new RuntimeException(
"Sorry, the orderPreservingForwarder_v0_001-plugin supports " +
"a maximum of " +Integer.MAX_VALUE +" * MAX_PAYLOAD bytes " +
"before its internal sequence number counter reaches its " +
"limit. Given the current MAX_PAYLOAD of "
+anonNode.MAX_PAYLOAD +" bytes, the limit is "
+Util.humanReadableByteCount(((long)Integer.MAX_VALUE)*(long)anonNode.MAX_PAYLOAD, true)
+" and has now been reached. TODO: add overflow-handling ;)"
);
}
private MixPlugIn getThis() {
return this;
}
private class UserData extends UserAttachment {
public PriorityQueue<QueueElement<Request>> queuedRequests;
public int nextSequenceNumberSend = -1;
public int nextSequenceNumberReceive = 0;
public UserData(User owner) {
super(owner, getThis());
queuedRequests = new PriorityQueue<QueueElement<Request>>();
}
}
@Override
public void write(User user, byte[] data) {
anonNode.write(user, data);
}
@Override
public Reply addLayer4Header(Reply reply) {
// TODO: get port-header from layer 4
UserData userData = reply.getOwner().getAttachment(this, UserData.class);
assert userData != null;
if (userData.nextSequenceNumberSend == Integer.MAX_VALUE)
displayErrorMessage();
userData.nextSequenceNumberSend++;
byte[] sequenceNumberHeader = Util.intToByteArray(userData.nextSequenceNumberSend);
reply.setByteMessage(Util.concatArrays(sequenceNumberHeader, reply.getByteMessage()));
//System.out.println("" +this +": sending " +userData.nextSequenceNumberSend); // TODO: remove
//System.out.println("" +this +": sending (reply (mix)): " +Util.toHex(reply.getByteMessage())); // TODO: remove
return reply;
}
@Override
public int getSizeOfLayer4Header() {
return 4; // seq.-header is 4 bytes long (an int value)
}
@Override
public int getMaxSizeOfNextWrite() {
return layer3controller.getMaxSizeOfNextWrite() - getSizeOfLayer4Header();
}
@Override
public int getMaxSizeOfNextRead() {
return layer3controller.getMaxSizeOfNextRead() - getSizeOfLayer4Header();
}
}