Add some tests of conversion and client-server comm.
continuous-integration/drone/push Build is passing Details

This commit is contained in:
Mark Nellemann 2023-07-18 15:21:52 +02:00
parent 8d6c7f8140
commit 5b7fec6033
12 changed files with 253 additions and 37 deletions

View File

@ -25,12 +25,13 @@ public class Statistics {
private final int MAX_TICKS_AVG = 300; private final int MAX_TICKS_AVG = 300;
private final int LOG_AVG_MODULO = 30; private final int LOG_AVG_MODULO = 30;
private final DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss").withZone(ZoneId.systemDefault()); private final DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss").withZone(ZoneId.systemDefault());
private long packetsTransferred, packetsTransferredTotal = 0; private long packetsTransferred;
private long packetsTransferredTotal = 0;
private long bytesTransferred, bytesTransferredTotal = 0; private long bytesTransferred, bytesTransferredTotal = 0;
private long bytesPerSec; private long bytesPerSec;
private long packetsPerSec; private long packetsPerSec;
private long packetsUnacked = 0; private long packetsUnacked = 0;
private int tickItererations = 0; private int tickIterations = 0;
private int tickTotal = 0; private int tickTotal = 0;
private final long[] bytesPerSecAvgTmp = new long[MAX_TICKS_AVG]; private final long[] bytesPerSecAvgTmp = new long[MAX_TICKS_AVG];
@ -52,8 +53,8 @@ public class Statistics {
// Because we do this every second ... // Because we do this every second ...
bytesPerSec = bytesTransferred; bytesPerSec = bytesTransferred;
packetsPerSec = packetsTransferred; packetsPerSec = packetsTransferred;
bytesPerSecAvgTmp[tickItererations] = bytesTransferred; bytesPerSecAvgTmp[tickIterations] = bytesTransferred;
packetsPerSecAvgTmp[tickItererations] = packetsTransferred; packetsPerSecAvgTmp[tickIterations] = packetsTransferred;
timestamp1 = timestamp2; timestamp1 = timestamp2;
printStatus(); printStatus();
@ -61,11 +62,11 @@ public class Statistics {
bytesTransferred = 0; bytesTransferred = 0;
packetsTransferred = 0; packetsTransferred = 0;
if(++tickItererations >= MAX_TICKS_AVG) { if(++tickIterations >= MAX_TICKS_AVG) {
tickItererations = 0; tickIterations = 0;
} }
if(tickItererations % LOG_AVG_MODULO == 0) { if(tickIterations % LOG_AVG_MODULO == 0) {
printAverage(); printAverage();
} }
tickTotal++; tickTotal++;
@ -78,7 +79,6 @@ public class Statistics {
public void printStatus() { public void printStatus() {
System.out.printf("%-19s - Status: %10d pkt/s %14d B/s %12d KB/s %8d MB/s\n", formatter.format(Instant.now()), packetsPerSec, bytesPerSec, bytesPerSec/1_000, bytesPerSec/1_000_000); System.out.printf("%-19s - Status: %10d pkt/s %14d B/s %12d KB/s %8d MB/s\n", formatter.format(Instant.now()), packetsPerSec, bytesPerSec, bytesPerSec/1_000, bytesPerSec/1_000_000);
} }
@ -122,6 +122,9 @@ public class Statistics {
return packetsTransferredTotal; return packetsTransferredTotal;
} }
public long getBytesTransferredTotal() {
return bytesTransferredTotal;
}
public int getRuntime() { public int getRuntime() {
return tickTotal; return tickTotal;
@ -157,5 +160,4 @@ public class Statistics {
return avg; return avg;
} }
} }

View File

@ -122,4 +122,9 @@ public class TcpClient {
statistics.printSummary(); statistics.printSummary();
} }
public Statistics getStatistics() {
return statistics;
}
} }

View File

@ -17,6 +17,8 @@ public class TcpServer extends Thread {
private DataInputStream in; private DataInputStream in;
private DataOutputStream out; private DataOutputStream out;
private byte[] inBuffer; private byte[] inBuffer;
private boolean runThread = true;
private boolean runSession = true;
public TcpServer(int port) throws IOException { public TcpServer(int port) throws IOException {
@ -28,7 +30,7 @@ public class TcpServer extends Thread {
public void run() { public void run() {
try { try {
while (true) { while (runThread) {
socket = new ServerSocket(port); socket = new ServerSocket(port);
socket.setSoTimeout(0); // Wait indefinitely socket.setSoTimeout(0); // Wait indefinitely
inBuffer = new byte[Payload.DEFAULT_LENGTH]; inBuffer = new byte[Payload.DEFAULT_LENGTH];
@ -45,8 +47,8 @@ public class TcpServer extends Thread {
public void session() throws IOException { public void session() throws IOException {
Statistics statistics = new Statistics(); Statistics statistics = new Statistics();
boolean running = true;
boolean ackEnd = false; boolean ackEnd = false;
runSession = true;
Socket server = socket.accept(); Socket server = socket.accept();
InetAddress address = socket.getInetAddress(); InetAddress address = socket.getInetAddress();
@ -54,7 +56,7 @@ public class TcpServer extends Thread {
in = new DataInputStream(server.getInputStream()); in = new DataInputStream(server.getInputStream());
out = new DataOutputStream(server.getOutputStream()); out = new DataOutputStream(server.getOutputStream());
while (running) { while (runSession) {
Payload payload = receive(); Payload payload = receive();
statistics.transferPacket(); statistics.transferPacket();
@ -78,7 +80,7 @@ public class TcpServer extends Thread {
statistics.tick(); statistics.tick();
if(ackEnd) { if(ackEnd) {
running = false; runSession = false;
statistics.printAverage(); statistics.printAverage();
statistics.printSummary(); statistics.printSummary();
} }
@ -97,4 +99,10 @@ public class TcpServer extends Thread {
return new Payload(inBuffer); return new Payload(inBuffer);
} }
public void finish() {
runThread = false;
runSession = false;
}
} }

View File

@ -29,8 +29,7 @@ public class TimeSuffixConverter implements CommandLine.ITypeConverter<Integer>
seconds = number * 60 * 60; seconds = number * 60 * 60;
break; break;
default: default:
System.err.println("Unknown suffix: " + suffix); throw new IllegalArgumentException("Unknown suffix: " + suffix);
seconds = number;
} }
} else { } else {
seconds = number; seconds = number;

View File

@ -118,7 +118,6 @@ public class UdpClient {
// Send end // Send end
//Thread.sleep(100);
payload = new Payload(PayloadType.END.getValue(), length, sequence++, packets); payload = new Payload(PayloadType.END.getValue(), length, sequence++, packets);
send(payload); send(payload);
payload = receive(); payload = receive();
@ -134,4 +133,9 @@ public class UdpClient {
statistics.printSummary(); statistics.printSummary();
} }
public Statistics getStatistics() {
return statistics;
}
} }

View File

@ -32,6 +32,9 @@ public class UdpServer extends Thread {
private DatagramSocket socket; private DatagramSocket socket;
private byte[] inBuffer; private byte[] inBuffer;
private boolean runThread = true;
private boolean runSession = true;
public UdpServer(int port) { public UdpServer(int port) {
log.info("UdpServer()"); log.info("UdpServer()");
@ -41,7 +44,7 @@ public class UdpServer extends Thread {
public void run() { public void run() {
try { try {
while (true) { while (runThread) {
inBuffer = new byte[Payload.DEFAULT_LENGTH]; inBuffer = new byte[Payload.DEFAULT_LENGTH];
socket = new DatagramSocket(port); socket = new DatagramSocket(port);
session(); session();
@ -57,10 +60,10 @@ public class UdpServer extends Thread {
public void session() throws IOException { public void session() throws IOException {
Statistics statistics = new Statistics(); Statistics statistics = new Statistics();
boolean running = true;
boolean ackEnd = false; boolean ackEnd = false;
runSession = true;
while (running) { while (runSession) {
DatagramPacket packet = new DatagramPacket(inBuffer, inBuffer.length); DatagramPacket packet = new DatagramPacket(inBuffer, inBuffer.length);
socket.receive(packet); socket.receive(packet);
@ -91,7 +94,7 @@ public class UdpServer extends Thread {
statistics.tick(); statistics.tick();
if(ackEnd) { if(ackEnd) {
running = false; runSession = false;
statistics.printAverage(); statistics.printAverage();
statistics.printSummary(); statistics.printSummary();
} }
@ -101,4 +104,11 @@ public class UdpServer extends Thread {
} }
public void finish() {
runThread = false;
runSession = false;
}
} }

View File

@ -6,16 +6,17 @@ import java.util.Locale;
import java.util.regex.Matcher; import java.util.regex.Matcher;
import java.util.regex.Pattern; import java.util.regex.Pattern;
public class UnitSuffixConverter implements CommandLine.ITypeConverter<Integer> { public class UnitSuffixConverter implements CommandLine.ITypeConverter<Long> {
final private Pattern pattern = Pattern.compile("(\\d+)([kmg])?b?", Pattern.CASE_INSENSITIVE); final private Pattern pattern = Pattern.compile("(\\d+)([kmg])?b?", Pattern.CASE_INSENSITIVE);
public Integer convert(String value) { public Long convert(String value) {
int bytes = 0;
long bytes = 0L;
Matcher matcher = pattern.matcher(value); Matcher matcher = pattern.matcher(value);
if (matcher.find()) { if (matcher.find()) {
int number = Integer.parseInt(matcher.group(1)); long number = Long.parseLong(matcher.group(1));
if(matcher.group(2) != null) { // We got the kilo, mega og giga suffix if(matcher.group(2) != null) { // We got the kilo, mega og giga suffix
String suffix = matcher.group(2); String suffix = matcher.group(2);
switch (suffix.toLowerCase(Locale.ROOT)) { switch (suffix.toLowerCase(Locale.ROOT)) {
@ -29,8 +30,7 @@ public class UnitSuffixConverter implements CommandLine.ITypeConverter<Integer>
bytes = number * 1024 * 1024 * 1024; bytes = number * 1024 * 1024 * 1024;
break; break;
default: default:
System.err.println("Unknown suffix: " + suffix); throw new IllegalArgumentException("Unknown suffix: " + suffix);
bytes = number;
} }
} else { } else {
bytes = number; bytes = number;

View File

@ -1,11 +0,0 @@
/*
* This Spock specification was generated by the Gradle 'init' task.
*/
package biz.nellemann.jnetperf
import spock.lang.Specification
class ApplicationTest extends Specification {
}

View File

@ -0,0 +1,45 @@
package biz.nellemann.jnetperf
import spock.lang.Shared
import spock.lang.Specification
class TcpClientServerTest extends Specification {
static final int port = 9876;
@Shared
TcpServer tcpServer = new TcpServer(port)
// run before every feature method
def setup() {
tcpServer.start();
}
// run after every feature method
def cleanup() {
tcpServer.finish()
}
// run before the first feature method
def setupSpec() {
}
// run after the last feature method
def cleanupSpec() {
}
def "test client to server communication"() {
setup:
TcpClient client = new TcpClient("localhost", port, 512, 100, 60)
when:
client.start()
then:
client.getStatistics().getPacketsTransferredTotal() == 102 // packets + handshake + end
client.getStatistics().getBytesTransferredTotal() == 52224
}
}

View File

@ -0,0 +1,38 @@
package biz.nellemann.jnetperf
import spock.lang.Shared
import spock.lang.Specification
class TimeSuffixConverterTest extends Specification {
@Shared
TimeSuffixConverter timeSuffixConverter = new TimeSuffixConverter();
def "test second to seconds"() {
when:
int seconds = timeSuffixConverter.convert("12s")
then:
seconds == 12;
}
def "test minute to seconds"() {
when:
int seconds = timeSuffixConverter.convert("120m")
then:
seconds == 7200;
}
def "test hour to seconds"() {
when:
int seconds = timeSuffixConverter.convert("48h")
then:
seconds == 172800;
}
}

View File

@ -0,0 +1,45 @@
package biz.nellemann.jnetperf
import spock.lang.Shared
import spock.lang.Specification
class UdpClientServerTest extends Specification {
static final int port = 9876;
@Shared
UdpServer udpServer = new UdpServer(port)
// run before every feature method
def setup() {
udpServer.start();
}
// run after every feature method
def cleanup() {
udpServer.finish()
}
// run before the first feature method
def setupSpec() {
}
// run after the last feature method
def cleanupSpec() {
}
def "test client to server communication"() {
setup:
UdpClient client = new UdpClient("localhost", port, 512, 100, 60)
when:
client.start()
then:
client.getStatistics().getPacketsTransferredTotal() == 102 // packets + handshake + end
client.getStatistics().getBytesTransferredTotal() == 53144 // TODO: Why is this larger than the TCP test ?
}
}

View File

@ -0,0 +1,71 @@
package biz.nellemann.jnetperf
import spock.lang.Shared
import spock.lang.Specification
class UnitSuffixConverterTest extends Specification {
@Shared
UnitSuffixConverter unitSuffixConverter = new UnitSuffixConverter();
def "test byte (b) to bytes"() {
when:
long bytes = unitSuffixConverter.convert("16b")
then:
bytes == 16;
}
def "test kilo (k) to bytes"() {
when:
long bytes = unitSuffixConverter.convert("2048k")
then:
bytes == 2097152;
}
def "test kilo (kb) to bytes"() {
when:
long bytes = unitSuffixConverter.convert("2048kb")
then:
bytes == 2097152;
}
def "test mega (m) to bytes"() {
when:
long bytes = unitSuffixConverter.convert("2m")
then:
bytes == 2097152;
}
def "test mega (mb) to bytes"() {
when:
long bytes = unitSuffixConverter.convert("2mb")
then:
bytes == 2097152;
}
def "test giga (g) to bytes"() {
when:
long bytes = unitSuffixConverter.convert("1g")
then:
bytes == 1073741824;
}
def "test giga (gb) to bytes"() {
when:
long bytes = unitSuffixConverter.convert("1gb")
then:
bytes == 1073741824;
}
}