jnetperf/src/main/java/biz/nellemann/jperf/UdpServer.java

143 lines
4.1 KiB
Java
Raw Normal View History

2023-06-20 07:34:33 +00:00
package biz.nellemann.jperf;
import java.io.IOException;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.InetAddress;
import java.net.SocketException;
2023-06-22 06:40:31 +00:00
import java.time.Duration;
import java.time.Instant;
2023-06-20 07:34:33 +00:00
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
public class UdpServer extends Thread {
final Logger log = LoggerFactory.getLogger(UdpServer.class);
2023-06-20 17:10:21 +00:00
private final DatagramSocket socket;
2023-06-20 07:34:33 +00:00
private byte[] buf = new byte[256];
2023-06-22 06:40:31 +00:00
long pktsReceived, pktsReceivedTotal = 0;
long bytesReceived, bytesReceivedTotal = 0;
long bytesPerSec, pktsPerSec = 0;
2023-06-21 14:39:37 +00:00
public UdpServer(int port) throws SocketException {
2023-06-22 06:40:31 +00:00
log.info("UdpServer()");
2023-06-21 14:39:37 +00:00
socket = new DatagramSocket(port);
2023-06-20 07:34:33 +00:00
}
public void run() {
2023-06-20 17:10:21 +00:00
2023-06-22 06:40:31 +00:00
boolean running = true;
2023-06-20 07:34:33 +00:00
try {
2023-06-20 17:10:21 +00:00
2023-06-20 07:34:33 +00:00
while (running) {
2023-06-22 06:40:31 +00:00
session();
}
2023-06-20 07:34:33 +00:00
2023-06-22 06:40:31 +00:00
socket.close();
} catch(IOException e) {
log.error(e.getMessage());
}
2023-06-20 07:34:33 +00:00
2023-06-22 06:40:31 +00:00
}
2023-06-20 07:34:33 +00:00
2023-06-22 06:40:31 +00:00
public void printStatistics() {
// Because we do this every second ...
bytesPerSec = bytesReceived;
pktsPerSec = pktsReceived;
2023-06-20 07:34:33 +00:00
2023-06-22 06:40:31 +00:00
System.out.printf("%s recv: %d pkt/s\t %d B/s\t %d KB/s\t %d MB/s\n", Instant.now().toString(), pktsPerSec, bytesPerSec, bytesPerSec/1_000, bytesPerSec/1_000_000);
pktsReceived = 0;
bytesReceived = 0;
}
2023-06-20 17:10:21 +00:00
2023-06-22 06:40:31 +00:00
public void printSummary() {
System.out.printf("%s recv: %d pkts\t %d B\t %d KB\t %d MB\n", Instant.now().toString(), pktsReceivedTotal, bytesReceivedTotal, bytesReceivedTotal/1_000, bytesReceivedTotal/1_000_000);
}
2023-06-20 17:10:21 +00:00
2023-06-20 07:34:33 +00:00
2023-06-22 06:40:31 +00:00
public void session() throws IOException {
2023-06-20 07:34:33 +00:00
2023-06-22 06:40:31 +00:00
boolean running = true;
boolean ackEnd = false;
long thisSequence, lastSequence = 0;
Instant startInstant = Instant.now();
Instant checkInstant;
while (running) {
DatagramPacket packet = new DatagramPacket(buf, buf.length);
socket.receive(packet);
InetAddress address = packet.getAddress();
int port = packet.getPort();
Datagram datagram = new Datagram(buf);
thisSequence = datagram.getCurPkt();
if(datagram.getType() == DataType.HANDSHAKE.getValue()) {
log.info("Handshake from ... {}, length: {}", address, datagram.getLength());
// Setup to receive larger datagrams
buf = new byte[datagram.getLength()];
2023-06-20 07:34:33 +00:00
2023-06-22 06:40:31 +00:00
// Send ACK
Datagram responseDatagram = new Datagram(DataType.ACK.getValue(), 32, datagram.getCurPkt(), 1);
packet = new DatagramPacket(responseDatagram.getPayload(), responseDatagram.getLength(), address, port);
socket.send(packet);
}
if(datagram.getType() == DataType.DATA.getValue()) {
bytesReceived += datagram.getLength();
bytesReceivedTotal += datagram.getLength();
if(thisSequence == lastSequence + 1) {
//log.info("Data .... size: {}, sequence: {}", datagram.getLength(), thisSequence);
} else {
//log.warn("Data .... out of sequence: {} vs {}", thisSequence, lastSequence);
2023-06-20 07:34:33 +00:00
}
2023-06-22 06:40:31 +00:00
}
2023-06-20 07:34:33 +00:00
2023-06-22 06:40:31 +00:00
if(datagram.getType() == DataType.END.getValue()) {
ackEnd = true;
2023-06-20 07:34:33 +00:00
}
2023-06-22 06:40:31 +00:00
// Every second
checkInstant = Instant.now();
if(Duration.between(startInstant, checkInstant).toSeconds() >= 1) {
printStatistics();
startInstant = checkInstant;
}
2023-06-20 07:34:33 +00:00
2023-06-22 06:40:31 +00:00
if(ackEnd && pktsReceivedTotal > datagram.getMaxPkt()) {
// Send ACK
Datagram responseDatagram = new Datagram(DataType.ACK.getValue(), 32, datagram.getCurPkt(), 1);
packet = new DatagramPacket(responseDatagram.getPayload(), responseDatagram.getLength(), address, port);
socket.send(packet);
2023-06-20 07:34:33 +00:00
2023-06-22 06:40:31 +00:00
printSummary();
running = false;
}
2023-06-20 07:34:33 +00:00
2023-06-22 06:40:31 +00:00
lastSequence = thisSequence;
pktsReceived++;
pktsReceivedTotal++;
2023-06-20 07:34:33 +00:00
2023-06-22 06:40:31 +00:00
}
2023-06-20 07:34:33 +00:00
2023-06-22 06:40:31 +00:00
}
2023-06-20 07:34:33 +00:00
}