KademliaDHT/src/kademlia/core/KadServer.java

316 lines
9.4 KiB
Java
Raw Normal View History

2014-02-18 20:37:07 +00:00
package kademlia.core;
import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.DataInputStream;
import java.io.DataOutputStream;
import java.io.IOException;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.SocketException;
import java.util.HashMap;
import java.util.Map;
2014-02-18 20:37:07 +00:00
import java.util.Random;
import java.util.Timer;
import java.util.TimerTask;
import kademlia.message.Message;
import kademlia.message.MessageFactory;
import kademlia.node.Node;
import kademlia.message.Receiver;
2014-02-18 20:37:07 +00:00
/**
* The server that handles sending and receiving messages between nodes on the Kad Network
*
* @author Joshua Kissoon
* @created 20140215
*/
2014-02-18 20:37:07 +00:00
public class KadServer
{
/* Maximum size of a Datagram Packet */
2014-02-18 20:37:07 +00:00
private static final int DATAGRAM_BUFFER_SIZE = 64 * 1024; // 64KB
/* Basic Kad Objects */
private final transient KadConfiguration config;
2014-02-18 20:37:07 +00:00
/* Server Objects */
private final int udpPort;
private final DatagramSocket socket;
private boolean isRunning;
private final Map<Integer, Receiver> receivers;
2014-02-18 20:37:07 +00:00
private final Timer timer; // Schedule future tasks
private final Map<Integer, TimerTask> tasks; // Keep track of scheduled tasks
2014-02-25 08:12:08 +00:00
private final Node localNode;
2014-02-18 20:37:07 +00:00
/* Factories */
private final MessageFactory messageFactory;
{
isRunning = true;
this.tasks = new HashMap<>();
this.receivers = new HashMap<>();
this.timer = new Timer(true);
2014-02-18 20:37:07 +00:00
}
/**
* Initialize our KadServer
*
* @param udpPort The port to listen on
* @param mFactory Factory used to create messages
* @param localNode Local node on which this server runs on
* @param config
*
* @throws java.net.SocketException
*/
public KadServer(int udpPort, MessageFactory mFactory, Node localNode, KadConfiguration config) throws SocketException
2014-02-18 20:37:07 +00:00
{
this.udpPort = udpPort;
this.config = config;
2014-02-18 20:37:07 +00:00
this.socket = new DatagramSocket(udpPort);
2014-02-25 08:12:08 +00:00
this.localNode = localNode;
2014-02-18 20:37:07 +00:00
this.messageFactory = mFactory;
/* Start listening for incoming requests in a new thread */
this.startListener();
}
/**
* Starts the listener to listen for incoming messages
*/
private void startListener()
{
new Thread()
{
@Override
public void run()
{
listen();
}
}.start();
}
/**
* Sends a message
*
* @param msg The message to send
* @param to The node to send the message to
* @param recv The receiver to handle the response message
*
* @return Integer The communication ID of this message
*
2014-02-18 20:37:07 +00:00
* @throws IOException
*/
public synchronized int sendMessage(Node to, Message msg, Receiver recv) throws IOException
2014-02-18 20:37:07 +00:00
{
if (!isRunning)
{
throw new IllegalStateException("Kad Server is not running.");
}
/* Generate a random communication ID */
int comm = new Random().nextInt();
2014-02-18 20:37:07 +00:00
/* If we have a receiver */
if (recv != null)
{
/* Setup the receiver to handle message response */
//System.out.println(this.localNode + " Putting Receiver for comm: " + comm + " Receiver: " + recv);
receivers.put(comm, recv);
TimerTask task = new TimeoutTask(comm, recv);
timer.schedule(task, this.config.responseTimeout());
tasks.put(comm, task);
}
2014-02-18 20:37:07 +00:00
/* Send the message */
sendMessage(to, msg, comm);
return comm;
2014-02-18 20:37:07 +00:00
}
/**
* Method called to reply to a message received
*
* @param to The Node to send the reply to
* @param msg The reply message
* @param comm The communication ID - the one received
*
* @throws java.io.IOException
*/
public synchronized void reply(Node to, Message msg, int comm) throws IOException
2014-02-18 20:37:07 +00:00
{
if (!isRunning)
{
throw new IllegalStateException("Kad Server is not running.");
}
sendMessage(to, msg, comm);
}
/**
* Internal sendMessage method called by the public sendMessage method after a communicationId is generated
*/
2014-02-18 20:37:07 +00:00
private void sendMessage(Node to, Message msg, int comm) throws IOException
{
/* Use a try-with resource to auto-close streams after usage */
try (ByteArrayOutputStream bout = new ByteArrayOutputStream(); DataOutputStream dout = new DataOutputStream(bout);)
{
/* Setup the message for transmission */
dout.writeInt(comm);
dout.writeByte(msg.code());
msg.toStream(dout);
dout.close();
2014-02-18 20:37:07 +00:00
byte[] data = bout.toByteArray();
2014-02-18 20:37:07 +00:00
if (data.length > DATAGRAM_BUFFER_SIZE)
{
throw new IOException("Message is too big");
}
2014-02-18 20:37:07 +00:00
/* Everything is good, now create the packet and send it */
DatagramPacket pkt = new DatagramPacket(data, 0, data.length);
pkt.setSocketAddress(to.getSocketAddress());
socket.send(pkt);
}
2014-02-18 20:37:07 +00:00
}
/**
* Listen for incoming messages in a separate thread
*/
private void listen()
{
try
{
while (isRunning)
{
try
{
/* Wait for a packet */
byte[] buffer = new byte[DATAGRAM_BUFFER_SIZE];
DatagramPacket packet = new DatagramPacket(buffer, buffer.length);
socket.receive(packet);
/* We've received a packet, now handle it */
try (ByteArrayInputStream bin = new ByteArrayInputStream(packet.getData(), packet.getOffset(), packet.getLength());
DataInputStream din = new DataInputStream(bin);)
{
2014-02-18 20:37:07 +00:00
/* Read in the conversation Id to know which handler to handle this response */
int comm = din.readInt();
byte messCode = din.readByte();
2014-02-18 20:37:07 +00:00
Message msg = messageFactory.createMessage(messCode, din);
din.close();
2014-02-18 20:37:07 +00:00
//System.out.println(this.localNode.getNodeId() + " Message Received: [Comm: " + comm + "] " + msg);
/* Get a receiver for this message */
Receiver receiver;
if (this.receivers.containsKey(comm))
{
/* If there is a reciever in the receivers to handle this */
synchronized (this)
{
receiver = this.receivers.remove(comm);
TimerTask task = (TimerTask) tasks.remove(comm);
task.cancel();
}
}
else
{
/* There is currently no receivers, try to get one */
receiver = messageFactory.createReceiver(messCode, this);
}
2014-02-18 20:37:07 +00:00
/* Invoke the receiver */
if (receiver != null)
{
receiver.receive(msg, comm);
}
2014-02-18 20:37:07 +00:00
}
}
catch (IOException e)
{
this.isRunning = false;
System.err.println("Server ran into a problem in listener method. Message: " + e.getMessage());
2014-02-18 20:37:07 +00:00
}
}
}
finally
{
if (!socket.isClosed())
{
socket.close();
}
this.isRunning = false;
}
}
/**
* Remove a conversation receiver
*
* @param comm The id of this conversation
*/
private synchronized void unregister(int comm)
{
receivers.remove(comm);
this.tasks.remove(comm);
}
/**
* Stops listening and shuts down the server
*/
public void shutdown()
{
this.isRunning = false;
this.socket.close();
timer.cancel();
2014-02-18 20:37:07 +00:00
}
/**
* Task that gets called by a separate thread if a timeout for a receiver occurs.
* When a reply arrives this task must be canceled using the <code>cancel()</code>
2014-02-18 20:37:07 +00:00
* method inherited from <code>TimerTask</code>. In this case the caller is
* responsible for removing the task from the <code>tasks</code> map.
* */
class TimeoutTask extends TimerTask
{
private final int comm;
private final Receiver recv;
public TimeoutTask(int comm, Receiver recv)
{
this.comm = comm;
this.recv = recv;
}
@Override
public void run()
{
try
{
unregister(comm);
recv.timeout(comm);
}
catch (IOException e)
{
System.err.println("Cannot unregister a receiver. Message: " + e.getMessage());
2014-02-18 20:37:07 +00:00
}
}
}
public void printReceivers()
{
for (Integer r : this.receivers.keySet())
{
System.out.println("Receiver for comm: " + r + "; Receiver: " + this.receivers.get(r));
}
}
2014-02-18 20:37:07 +00:00
}