Compare commits
9 Commits
fc0981852a
...
ce814df02e
Author | SHA1 | Date |
---|---|---|
mike | ce814df02e | |
mike | 078a7f2b7a | |
Brett | 213bbd3f68 | |
Brett | 92f2c9f309 | |
Brett | 10da34999b | |
Brett | 1842816ba7 | |
Brett | 941aa58576 | |
Brett | c106cca6b1 | |
Brett | 24708a2fda |
Binary file not shown.
|
@ -3,10 +3,10 @@ package ca.cosc3p91.a4;
|
||||||
import ca.cosc3p91.a4.util.network.Client;
|
import ca.cosc3p91.a4.util.network.Client;
|
||||||
import java.io.*;
|
import java.io.*;
|
||||||
|
|
||||||
public class Main {
|
public class MainClient {
|
||||||
|
|
||||||
public static void main(String[] args) throws IOException {
|
public static void main(String[] args) throws IOException {
|
||||||
Client gameClient = new Client("localhost");
|
new Client("localhost");
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
|
@ -0,0 +1,11 @@
|
||||||
|
package ca.cosc3p91.a4;
|
||||||
|
|
||||||
|
import ca.cosc3p91.a4.util.network.Server;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
|
||||||
|
public class MainServer {
|
||||||
|
public static void main(String[] args) throws IOException, InterruptedException {
|
||||||
|
new Server();
|
||||||
|
}
|
||||||
|
}
|
|
@ -6,6 +6,7 @@ import ca.cosc3p91.a4.gameobjects.factory.InhabitantFactory;
|
||||||
import ca.cosc3p91.a4.player.*;
|
import ca.cosc3p91.a4.player.*;
|
||||||
import ca.cosc3p91.a4.userinterface.GameDisplay;
|
import ca.cosc3p91.a4.userinterface.GameDisplay;
|
||||||
import ca.cosc3p91.a4.util.ChallengeAdapter;
|
import ca.cosc3p91.a4.util.ChallengeAdapter;
|
||||||
|
import ca.cosc3p91.a4.util.network.Server;
|
||||||
|
|
||||||
import java.beans.XMLEncoder;
|
import java.beans.XMLEncoder;
|
||||||
import java.io.BufferedOutputStream;
|
import java.io.BufferedOutputStream;
|
||||||
|
@ -19,36 +20,31 @@ public class GameEngine {
|
||||||
public static final double IRON_FACTOR = 1;
|
public static final double IRON_FACTOR = 1;
|
||||||
public static final double WOOD_FACTOR = 0.1;
|
public static final double WOOD_FACTOR = 0.1;
|
||||||
|
|
||||||
private float pillageFactor = 0.5f;
|
|
||||||
|
|
||||||
private int currentTime;
|
private int currentTime;
|
||||||
|
|
||||||
private final Random random = new Random(System.nanoTime());
|
private final Random random = new Random(System.nanoTime());
|
||||||
|
|
||||||
public GameDisplay view;
|
public GameDisplay view = new GameDisplay();
|
||||||
|
|
||||||
public GameEngine() {
|
public GameEngine() {
|
||||||
}
|
}
|
||||||
|
|
||||||
public void attackVillage(Map attacking, Map defending) {
|
public boolean attackVillage(Map attacking, Map defending, boolean simulated, Server.ConnectedClient client) {
|
||||||
// int defenseiveCounter = 1;
|
int defenseiveCounter = 1;
|
||||||
// int inhabCounter = 0;
|
int inhabCounter = 0;
|
||||||
// for (Building b : map.contains)
|
for (Building b : defending.contains)
|
||||||
// if (b instanceof DefenseBuilding)
|
if (b instanceof DefenseBuilding)
|
||||||
// defenseiveCounter++;
|
defenseiveCounter++;
|
||||||
// for (Inhabitant i : map.inhabitants)
|
for (Inhabitant i : defending.inhabitants)
|
||||||
// if (i instanceof Infantry)
|
if (i instanceof Infantry)
|
||||||
// inhabCounter++;
|
inhabCounter++;
|
||||||
// pillageFactor = (float) inhabCounter / (float) defenseiveCounter;
|
float pillageFactor = (float) inhabCounter / (float) defenseiveCounter;
|
||||||
// if (pillageFactor < 0)
|
if (pillageFactor < 0)
|
||||||
// pillageFactor = 0;
|
pillageFactor = 0;
|
||||||
// if (pillageFactor > 1)
|
if (pillageFactor > 1)
|
||||||
// pillageFactor = 1;
|
pillageFactor = 1;
|
||||||
// this.map.getTownHall().addWood((int) (map.getTownHall().getCurrentWood() * pillageFactor));
|
|
||||||
// this.map.getTownHall().addIron((int) (map.getTownHall().getCurrentIron() * pillageFactor));
|
|
||||||
// this.map.getTownHall().addGold((int) (map.getTownHall().getCurrentGold() * pillageFactor));
|
|
||||||
ChallengeAdapter adapter = new ChallengeAdapter(attacking);
|
ChallengeAdapter adapter = new ChallengeAdapter(attacking);
|
||||||
adapter.attack(defending);
|
return adapter.attack(defending, simulated, pillageFactor, client);
|
||||||
}
|
}
|
||||||
|
|
||||||
public Map generateInitialMap(){
|
public Map generateInitialMap(){
|
||||||
|
@ -147,23 +143,27 @@ public class GameEngine {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public synchronized boolean build (Map map, String buildingArg) {
|
public synchronized boolean build (Map map, String buildingArg) throws BuildingErrorException {
|
||||||
BuildingFactory bfactory = new BuildingFactory();
|
BuildingFactory bfactory = new BuildingFactory();
|
||||||
Building type = bfactory.getBuilding(buildingArg);
|
Building type = bfactory.getBuilding(buildingArg);
|
||||||
|
if (type == null)
|
||||||
|
throw new BuildingErrorException("Invalid building type!");
|
||||||
return map.build(new Tile(), type);
|
return map.build(new Tile(), type);
|
||||||
}
|
}
|
||||||
|
|
||||||
public boolean train (Map map, String inhabitantArgs) {
|
public boolean train (Map map, String inhabitantArgs) throws TrainingErrorException {
|
||||||
InhabitantFactory ifactory = new InhabitantFactory();
|
InhabitantFactory ifactory = new InhabitantFactory();
|
||||||
Inhabitant type = ifactory.getInhabitant(inhabitantArgs);
|
Inhabitant type = ifactory.getInhabitant(inhabitantArgs);
|
||||||
|
if (type == null)
|
||||||
|
throw new TrainingErrorException("Invalid training type!");
|
||||||
return map.train(type);
|
return map.train(type);
|
||||||
}
|
}
|
||||||
|
|
||||||
public synchronized boolean upgradeBuilding (Map map, int buildingIndex) {
|
public synchronized boolean upgradeBuilding (Map map, int buildingIndex) throws UpgradingErrorException {
|
||||||
return map.upgradeBuilding(buildingIndex);
|
return map.upgradeBuilding(buildingIndex);
|
||||||
}
|
}
|
||||||
|
|
||||||
public boolean upgradeInhabitant (Map map, int inhabitantIndex) {
|
public synchronized boolean upgradeInhabitant (Map map, int inhabitantIndex) {
|
||||||
return map.upgradeInhabitant(inhabitantIndex);
|
return map.upgradeInhabitant(inhabitantIndex);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -279,4 +279,22 @@ public class GameEngine {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static class BuildingErrorException extends Exception {
|
||||||
|
public BuildingErrorException(String message){
|
||||||
|
super(message);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static class TrainingErrorException extends Exception {
|
||||||
|
public TrainingErrorException(String message){
|
||||||
|
super(message);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static class UpgradingErrorException extends Exception {
|
||||||
|
public UpgradingErrorException(String message){
|
||||||
|
super(message);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -54,17 +54,20 @@ public class Map implements Serializable {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
public boolean upgradeBuilding(int buildingIndex) {
|
public boolean upgradeBuilding(int buildingIndex) throws GameEngine.UpgradingErrorException {
|
||||||
|
|
||||||
if (buildingIndex >= contains.size() || buildingIndex < 0) return false;
|
if (buildingIndex >= contains.size() || buildingIndex < 0)
|
||||||
|
return false;
|
||||||
|
|
||||||
Building b = contains.get(buildingIndex);
|
Building b = contains.get(buildingIndex);
|
||||||
|
|
||||||
int currentLevel = b.getLevel();
|
int currentLevel = b.getLevel();
|
||||||
CasaDeNarino hall = getTownHall();
|
CasaDeNarino hall = getTownHall();
|
||||||
|
|
||||||
if (currentLevel >= 2) return false;
|
if (currentLevel >= 2)
|
||||||
else if (b instanceof Farm) return true;
|
return false;
|
||||||
|
else if (b instanceof Farm)
|
||||||
|
return true;
|
||||||
|
|
||||||
int goldCost = b.getUpgradeStage().getCost(SaulGoodMine.resource);
|
int goldCost = b.getUpgradeStage().getCost(SaulGoodMine.resource);
|
||||||
int ironCost = b.getUpgradeStage().getCost(IronMine.resource);
|
int ironCost = b.getUpgradeStage().getCost(IronMine.resource);
|
||||||
|
@ -86,7 +89,8 @@ public class Map implements Serializable {
|
||||||
} else {
|
} else {
|
||||||
b.upgrade(VillageHallStages.villageStages[currentLevel + 1]);
|
b.upgrade(VillageHallStages.villageStages[currentLevel + 1]);
|
||||||
}
|
}
|
||||||
} else return false;
|
} else
|
||||||
|
return false;
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
|
@ -9,10 +9,11 @@ import java.io.BufferedReader;
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.io.InputStream;
|
import java.io.InputStream;
|
||||||
import java.io.InputStreamReader;
|
import java.io.InputStreamReader;
|
||||||
|
import java.util.ArrayList;
|
||||||
import java.util.Scanner;
|
import java.util.Scanner;
|
||||||
|
|
||||||
public class GameDisplay {
|
public class GameDisplay {
|
||||||
private BufferedReader reader;
|
private final BufferedReader reader;
|
||||||
private String input;
|
private String input;
|
||||||
|
|
||||||
public GameDisplay() {
|
public GameDisplay() {
|
||||||
|
@ -30,7 +31,7 @@ public class GameDisplay {
|
||||||
System.out.println("\t->" + input + '\n');
|
System.out.println("\t->" + input + '\n');
|
||||||
}
|
}
|
||||||
|
|
||||||
public void printVillageState(Map map, String displayName) {
|
public ArrayList<String> getVillageStateTable(Map map, String displayName){
|
||||||
Print resourcesPrinter = new Print(displayName, 2);
|
Print resourcesPrinter = new Print(displayName, 2);
|
||||||
|
|
||||||
resourcesPrinter.addColumn(new Print.Column("Resource Type"));
|
resourcesPrinter.addColumn(new Print.Column("Resource Type"));
|
||||||
|
@ -52,7 +53,7 @@ public class GameDisplay {
|
||||||
Integer.toString(map.getTownHall().getGoldCapacity()),
|
Integer.toString(map.getTownHall().getGoldCapacity()),
|
||||||
Integer.toString(map.getTownHall().getCurrentGold())));
|
Integer.toString(map.getTownHall().getCurrentGold())));
|
||||||
|
|
||||||
Print.print(resourcesPrinter.createTable(true, false, true));
|
ArrayList<String> total = new ArrayList<>(resourcesPrinter.createTable(true, false, true));
|
||||||
|
|
||||||
Print buildingPrinter = new Print("Village Buildings", 2, resourcesPrinter.getWidth());
|
Print buildingPrinter = new Print("Village Buildings", 2, resourcesPrinter.getWidth());
|
||||||
buildingPrinter.addColumn(new Print.Column("Name"));
|
buildingPrinter.addColumn(new Print.Column("Name"));
|
||||||
|
@ -64,7 +65,7 @@ public class GameDisplay {
|
||||||
Integer.toString(b.getLevel() + 1),
|
Integer.toString(b.getLevel() + 1),
|
||||||
Integer.toString(b.getHealth())));
|
Integer.toString(b.getHealth())));
|
||||||
|
|
||||||
Print.print(buildingPrinter.createTable(true, false, true));
|
total.addAll(buildingPrinter.createTable(true, false, true));
|
||||||
|
|
||||||
Print inhabs = new Print("Village Inhabitants", 2, buildingPrinter.getWidth());
|
Print inhabs = new Print("Village Inhabitants", 2, buildingPrinter.getWidth());
|
||||||
inhabs.addColumn(new Print.Column("Name"));
|
inhabs.addColumn(new Print.Column("Name"));
|
||||||
|
@ -73,17 +74,29 @@ public class GameDisplay {
|
||||||
for (Inhabitant i : map.inhabitants)
|
for (Inhabitant i : map.inhabitants)
|
||||||
inhabs.addRow(new Print.Row(i.getClass().getSimpleName(), Integer.toString(i.getLevel() + 1)));
|
inhabs.addRow(new Print.Row(i.getClass().getSimpleName(), Integer.toString(i.getLevel() + 1)));
|
||||||
|
|
||||||
Print.print(inhabs.createTable(true, true, true));
|
total.addAll(inhabs.createTable(true, true, true));
|
||||||
|
System.out.println(buildingPrinter.getWidth());
|
||||||
|
System.out.println(resourcesPrinter.getWidth());
|
||||||
|
System.out.println(inhabs.getWidth());
|
||||||
|
return total;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void printVillageState(Map map, String displayName) {
|
||||||
|
Print.print(getVillageStateTable(map, displayName));
|
||||||
}
|
}
|
||||||
|
|
||||||
public void printGameMenu() {
|
public void printGameMenu() {
|
||||||
System.out.println("\n~ Player Options:\n" +
|
System.out.println("\n\033[36m~ Player Options:\n" +
|
||||||
"1. Build {command: '1 <building name>'}\n" +
|
"1. Build {command: '1 <building name>'}\n" +
|
||||||
"2. Train inhabitants {command: '2 <unit name>'}\n"+
|
"2. Train inhabitants {command: '2 <unit name>'}\n"+
|
||||||
"3. Upgrade {command: '3 i<index>'} / {command: '3 b<index>'}\n"+
|
"3. Upgrade {command: '3 i<index>'} / {command: '3 b<index>'}\n"+
|
||||||
"4. Explore\n"+
|
"4. Explore Player Villages\n"+
|
||||||
"5. Print Village Stats\n"+
|
"5. Print Village Stats\n"+
|
||||||
"6. Quit\n" +
|
"6. Quit\n" +
|
||||||
"7. Attack last explored\n");
|
"7. Attack last explored/generated\n" +
|
||||||
|
"8. Generate Village\n" +
|
||||||
|
"9. Generate and Test Army\n" +
|
||||||
|
"0. Village Testing" +
|
||||||
|
"\033[0m\n");
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -3,9 +3,11 @@ package ca.cosc3p91.a4.util;
|
||||||
import ChallengeDecision.*;
|
import ChallengeDecision.*;
|
||||||
import ca.cosc3p91.a4.game.Map;
|
import ca.cosc3p91.a4.game.Map;
|
||||||
import ca.cosc3p91.a4.gameobjects.*;
|
import ca.cosc3p91.a4.gameobjects.*;
|
||||||
|
import ca.cosc3p91.a4.util.network.Server;
|
||||||
|
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
import java.util.concurrent.atomic.AtomicInteger;
|
||||||
|
|
||||||
public class ChallengeAdapter {
|
public class ChallengeAdapter {
|
||||||
|
|
||||||
|
@ -79,7 +81,7 @@ public class ChallengeAdapter {
|
||||||
this.map = map;
|
this.map = map;
|
||||||
}
|
}
|
||||||
|
|
||||||
public void attack(Map enemy){
|
public boolean attack(Map enemy, boolean simulated, float pillageFactor, Server.ConnectedClient client){
|
||||||
MapChallengeConverter enemyMap = new MapChallengeConverter(enemy);
|
MapChallengeConverter enemyMap = new MapChallengeConverter(enemy);
|
||||||
MapChallengeConverter ourMap = new MapChallengeConverter(this.map);
|
MapChallengeConverter ourMap = new MapChallengeConverter(this.map);
|
||||||
|
|
||||||
|
@ -116,28 +118,50 @@ public class ChallengeAdapter {
|
||||||
|
|
||||||
// if any fail to attack we need to pretend like it was one big attack that failed
|
// if any fail to attack we need to pretend like it was one big attack that failed
|
||||||
if (!goldResults.getChallengeWon() || !ironResults.getChallengeWon() || !woodResults.getChallengeWon())
|
if (!goldResults.getChallengeWon() || !ironResults.getChallengeWon() || !woodResults.getChallengeWon())
|
||||||
return;
|
return false;
|
||||||
|
|
||||||
System.out.println("We won gold: ");
|
System.out.print("We won gold: ");
|
||||||
goldResults.print();
|
goldResults.print();
|
||||||
System.out.println("We won iron: ");
|
System.out.print("We won iron: ");
|
||||||
ironResults.print();
|
ironResults.print();
|
||||||
System.out.println("We won wood: ");
|
System.out.print("We won wood: ");
|
||||||
woodResults.print();
|
woodResults.print();
|
||||||
|
|
||||||
CasaDeNarino th = map.getTownHall();
|
CasaDeNarino th = map.getTownHall();
|
||||||
|
CasaDeNarino eth = enemy.getTownHall();
|
||||||
|
|
||||||
|
AtomicInteger totalGold = new AtomicInteger();
|
||||||
|
AtomicInteger totalIron = new AtomicInteger();
|
||||||
|
AtomicInteger totalWood = new AtomicInteger();
|
||||||
|
|
||||||
|
if (!simulated) {
|
||||||
goldResults.getLoot().forEach(r -> {
|
goldResults.getLoot().forEach(r -> {
|
||||||
th.addGold((int)r.getProperty().doubleValue());
|
int gold = (int) (r.getProperty() * pillageFactor);
|
||||||
|
th.addGold(gold);
|
||||||
|
eth.addGold(-gold);
|
||||||
|
totalGold.addAndGet(gold);
|
||||||
});
|
});
|
||||||
|
|
||||||
ironResults.getLoot().forEach(r -> {
|
ironResults.getLoot().forEach(r -> {
|
||||||
th.addIron((int)r.getProperty().doubleValue());
|
int iron = (int) (r.getProperty() * pillageFactor);
|
||||||
|
th.addIron(iron);
|
||||||
|
eth.addIron(-iron);
|
||||||
|
totalIron.addAndGet(iron);
|
||||||
});
|
});
|
||||||
|
|
||||||
woodResults.getLoot().forEach(r -> {
|
woodResults.getLoot().forEach(r -> {
|
||||||
th.addWood((int)r.getProperty().doubleValue());
|
int wood = (int) (r.getProperty() * pillageFactor);
|
||||||
|
th.addWood(wood);
|
||||||
|
eth.addWood(-wood);
|
||||||
|
totalWood.addAndGet(wood);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
client.sendAndLogLn("You won gold: " + totalGold.get());
|
||||||
|
client.sendAndLogLn("You won iron: " + totalIron.get());
|
||||||
|
client.sendAndLogLn("You won wood: " + totalWood.get());
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -9,17 +9,24 @@ import java.net.DatagramPacket;
|
||||||
import java.net.DatagramSocket;
|
import java.net.DatagramSocket;
|
||||||
import java.net.InetAddress;
|
import java.net.InetAddress;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
|
import java.util.Collections;
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
public class Client implements Runnable {
|
public class Client implements Runnable {
|
||||||
private GameDisplay view = new GameDisplay();
|
private GameDisplay view = new GameDisplay();
|
||||||
|
|
||||||
private DatagramSocket clientSocket;
|
private DatagramSocket clientSocket;
|
||||||
private boolean running = true;
|
private volatile boolean running = true;
|
||||||
private Thread receiveThread;
|
private Thread receiveThread;
|
||||||
private final HashMap<Long, Message.Sent> sentMessages = new HashMap<>();
|
private final Map<Long, Message.Sent> sentMessages = Collections.synchronizedMap(new HashMap<>());
|
||||||
private int lastMessageID = 0;
|
private int lastMessageID = 0;
|
||||||
private final InetAddress serverAddress;
|
private final InetAddress serverAddress;
|
||||||
|
private String[] lineBuffer = new String[0];
|
||||||
|
private int expectedLines = 0;
|
||||||
|
private int currentLines = 0;
|
||||||
|
|
||||||
|
private long ourClientID = 0;
|
||||||
|
|
||||||
public Client(String address) throws IOException {
|
public Client(String address) throws IOException {
|
||||||
serverAddress = InetAddress.getByName(address);
|
serverAddress = InetAddress.getByName(address);
|
||||||
|
@ -27,35 +34,64 @@ public class Client implements Runnable {
|
||||||
receiveThread = new Thread(this);
|
receiveThread = new Thread(this);
|
||||||
receiveThread.start();
|
receiveThread.start();
|
||||||
|
|
||||||
sendMessage(new Message.Sent(PacketTable.CONNECT, 0, ++lastMessageID));
|
sendMessage(new Message.Sent(PacketIDs.CONNECT, ourClientID, ++lastMessageID));
|
||||||
|
|
||||||
|
view.printGameMenu();
|
||||||
|
|
||||||
while (running) {
|
while (running) {
|
||||||
String prompt;
|
String prompt;
|
||||||
if ((prompt = view.nextInput()) != null) {
|
if ((prompt = view.nextInput()) != null) {
|
||||||
if (prompt.trim().isEmpty())
|
if (prompt.trim().isEmpty())
|
||||||
continue;
|
continue;
|
||||||
if (prompt.charAt(0) == '6')
|
if (prompt.charAt(0) == '6') {
|
||||||
|
running = false;
|
||||||
break;
|
break;
|
||||||
|
}
|
||||||
|
view.printGameMenu();
|
||||||
|
String[] args = prompt.split(" ");
|
||||||
|
char c = prompt.charAt(0);
|
||||||
|
if (c > '0' && c < '4') {
|
||||||
|
if (args.length < 2) {
|
||||||
|
System.err.println("Args must include type!");
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
byte messageType;
|
byte messageType;
|
||||||
switch (prompt.charAt(0)) {
|
switch (c) {
|
||||||
case '1':
|
case '1':
|
||||||
messageType = PacketTable.BUILD;
|
messageType = PacketIDs.BUILD;
|
||||||
break;
|
break;
|
||||||
case '2':
|
case '2':
|
||||||
messageType = PacketTable.TRAIN;
|
messageType = PacketIDs.TRAIN;
|
||||||
break;
|
break;
|
||||||
case '3':
|
case '3':
|
||||||
messageType = PacketTable.UPGRADE;
|
messageType = PacketIDs.UPGRADE;
|
||||||
|
break;
|
||||||
|
case '4':
|
||||||
|
messageType = PacketIDs.EXPLORE;
|
||||||
|
break;
|
||||||
|
case '5':
|
||||||
|
messageType = PacketIDs.PRINT_MAP_DATA;
|
||||||
|
break;
|
||||||
|
case '7':
|
||||||
|
messageType = PacketIDs.ATTACK;
|
||||||
|
break;
|
||||||
|
case '8':
|
||||||
|
messageType = PacketIDs.GENERATE;
|
||||||
|
break;
|
||||||
|
case '9':
|
||||||
|
messageType = PacketIDs.TEST_ARMY;
|
||||||
|
break;
|
||||||
|
case '0':
|
||||||
|
messageType = PacketIDs.TEST_VILLAGE;
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
System.err.println("> Invalid command input!");
|
System.err.println("> Invalid command input!");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
Message.Sent buildMessage = new Message.Sent(messageType,0,++lastMessageID);
|
Message.Sent buildMessage = new Message.Sent(messageType,ourClientID,++lastMessageID);
|
||||||
buildMessage.getData().write(prompt.substring(1).getBytes());
|
buildMessage.getWriter().writeUTF(prompt.substring(1));
|
||||||
sendMessage(buildMessage);
|
sendMessage(buildMessage);
|
||||||
|
|
||||||
view.printGameMenu();
|
|
||||||
}
|
}
|
||||||
ArrayList<Long> removes = new ArrayList<>();
|
ArrayList<Long> removes = new ArrayList<>();
|
||||||
for (HashMap.Entry<Long, Message.Sent> message : sentMessages.entrySet()){
|
for (HashMap.Entry<Long, Message.Sent> message : sentMessages.entrySet()){
|
||||||
|
@ -85,10 +121,12 @@ public class Client implements Runnable {
|
||||||
long clientID = stream.readLong();
|
long clientID = stream.readLong();
|
||||||
long messageID = stream.readLong();
|
long messageID = stream.readLong();
|
||||||
|
|
||||||
System.out.println("Receiving message with ID " + messageID + " to client: " + clientID + " of type " + packetID);
|
System.out.println("Receiving message with ID " + messageID + " from server of type " + packetID + " our ClientID " + clientID + " / " + ourClientID);
|
||||||
|
|
||||||
switch (packetID) {
|
switch (packetID) {
|
||||||
case PacketTable.ACK:
|
case PacketIDs.ACK:
|
||||||
|
if (ourClientID == 0)
|
||||||
|
ourClientID = clientID;
|
||||||
Message.Sent message = sentMessages.get(messageID);
|
Message.Sent message = sentMessages.get(messageID);
|
||||||
if (message == null)
|
if (message == null)
|
||||||
throw new RuntimeException("Server acknowledged a message we never sent! (" + messageID + ")");
|
throw new RuntimeException("Server acknowledged a message we never sent! (" + messageID + ")");
|
||||||
|
@ -99,11 +137,32 @@ public class Client implements Runnable {
|
||||||
for (HashMap.Entry<Long, Message.Sent> ms : sentMessages.entrySet())
|
for (HashMap.Entry<Long, Message.Sent> ms : sentMessages.entrySet())
|
||||||
System.out.println("MessageID: " + ms.getKey());
|
System.out.println("MessageID: " + ms.getKey());
|
||||||
break;
|
break;
|
||||||
case PacketTable.DISCONNECT:
|
case PacketIDs.MESSAGE:
|
||||||
|
System.out.println("\033[93m" + stream.readUTF() + "\033[0m");
|
||||||
|
break;
|
||||||
|
case PacketIDs.BEGIN_MAP_DATA:
|
||||||
|
expectedLines = stream.readInt();
|
||||||
|
currentLines = 0;
|
||||||
|
lineBuffer = new String[expectedLines];
|
||||||
|
break;
|
||||||
|
case PacketIDs.MAP_LINE_DATA:
|
||||||
|
int lineNumber = stream.readInt();
|
||||||
|
lineBuffer[lineNumber] = stream.readUTF();
|
||||||
|
currentLines++;
|
||||||
|
if (currentLines >= expectedLines) {
|
||||||
|
for (String line : lineBuffer){
|
||||||
|
System.out.println("\033[92m" + line + "\033[0m");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case PacketIDs.DISCONNECT:
|
||||||
|
System.out.println("Disconnecting!");
|
||||||
running = false;
|
running = false;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
if (packetID != PacketIDs.ACK && packetID != PacketIDs.DISCONNECT){
|
||||||
|
sendMessage(new Message.Sent(PacketIDs.ACK, ourClientID, messageID));
|
||||||
|
}
|
||||||
} catch (Exception e){
|
} catch (Exception e){
|
||||||
e.printStackTrace();
|
e.printStackTrace();
|
||||||
}
|
}
|
||||||
|
@ -111,7 +170,7 @@ public class Client implements Runnable {
|
||||||
}
|
}
|
||||||
|
|
||||||
private void sendMessage(Message.Sent message){
|
private void sendMessage(Message.Sent message){
|
||||||
if (message.getPacketID() != PacketTable.ACK)
|
if (message.getPacketID() != PacketIDs.ACK)
|
||||||
sentMessages.put(message.getMessageID(), message);
|
sentMessages.put(message.getMessageID(), message);
|
||||||
byte[] data = message.getData().toByteArray();
|
byte[] data = message.getData().toByteArray();
|
||||||
DatagramPacket sendPacket = new DatagramPacket(data, data.length, serverAddress, Server.SERVER_PORT);
|
DatagramPacket sendPacket = new DatagramPacket(data, data.length, serverAddress, Server.SERVER_PORT);
|
||||||
|
|
|
@ -0,0 +1,41 @@
|
||||||
|
package ca.cosc3p91.a4.util.network;
|
||||||
|
|
||||||
|
public class PacketIDs {
|
||||||
|
|
||||||
|
// packetID -> byte defined in this file
|
||||||
|
// clientID -> long
|
||||||
|
// messageID -> long
|
||||||
|
|
||||||
|
// MESSAGE_HEADER (packetID, clientID, messageID)
|
||||||
|
|
||||||
|
// MESSAGE_HEADER, (clientID = 0 if connecting to server)
|
||||||
|
public static final byte CONNECT = 0x1;
|
||||||
|
// MESSAGE_HEADER
|
||||||
|
public static final byte DISCONNECT = 0x2;
|
||||||
|
// MESSAGE_HEADER
|
||||||
|
public static final byte ACK = 0x3;
|
||||||
|
// MESSAGE_HEADER, UTF8 String with length information (use DOS.writeUTF/DIS.readUTF)
|
||||||
|
public static final byte MESSAGE = 0x4;
|
||||||
|
// MESSAGE_HEADER, build
|
||||||
|
public static final byte BUILD = 0x5;
|
||||||
|
// MESSAGE_HEADER, train
|
||||||
|
public static final byte TRAIN = 0x6;
|
||||||
|
// MESSAGE_HEADER, upgrade
|
||||||
|
public static final byte UPGRADE = 0x7;
|
||||||
|
// MESSAGE_HEADER
|
||||||
|
public static final byte PRINT_MAP_DATA = 0x8; // client -> server only!
|
||||||
|
// MESSAGE_HEADER, line count
|
||||||
|
public static final byte BEGIN_MAP_DATA = 0x9; // server -> client
|
||||||
|
// MESSAGE_HEADER, line number (int), UTF8 String (the line)
|
||||||
|
public static final byte MAP_LINE_DATA = 0xA; // server -> client
|
||||||
|
// MESSAGE_HEADER
|
||||||
|
public static final byte EXPLORE = 0xB;
|
||||||
|
// MESSAGE_HEADER
|
||||||
|
public static final byte ATTACK = 0xC;
|
||||||
|
// MESSAGE_HEADER
|
||||||
|
public static final byte GENERATE = 0xD;
|
||||||
|
// MESSAGE_HEADER
|
||||||
|
public static final byte TEST_ARMY = 0xE;
|
||||||
|
// MESSAGE_HEADER
|
||||||
|
public static final byte TEST_VILLAGE = 0xF;
|
||||||
|
}
|
|
@ -1,28 +0,0 @@
|
||||||
package ca.cosc3p91.a4.util.network;
|
|
||||||
|
|
||||||
public class PacketTable {
|
|
||||||
|
|
||||||
// packetID -> byte defined in this file
|
|
||||||
// clientID -> long
|
|
||||||
// messageID -> long
|
|
||||||
|
|
||||||
// messageHeader (packetID, clientID, messageID)
|
|
||||||
|
|
||||||
// messageHeader, (clientID = 0 if connecting to server)
|
|
||||||
public static final byte CONNECT = 0x1;
|
|
||||||
// messageHeader
|
|
||||||
public static final byte DISCONNECT = 0x2;
|
|
||||||
// messageHeader
|
|
||||||
public static final byte ACK = 0x3;
|
|
||||||
// messageHeader, UTF8 String with length information (use DOS.writeUTF/DIS.readUTF)
|
|
||||||
public static final byte MESSAGE = 0x4;
|
|
||||||
// messageHeader, build
|
|
||||||
public static final byte BUILD = 0x5;
|
|
||||||
// messageHeader, train
|
|
||||||
public static final byte TRAIN = 0x6;
|
|
||||||
// messageHeader, upgrade
|
|
||||||
public static final byte UPGRADE = 0x7;
|
|
||||||
// messageHeader, serial packets with map info
|
|
||||||
public static final byte USER_MAP_DATA = 0x8;
|
|
||||||
|
|
||||||
}
|
|
|
@ -2,17 +2,15 @@ package ca.cosc3p91.a4.util.network;
|
||||||
|
|
||||||
import ca.cosc3p91.a4.game.GameEngine;
|
import ca.cosc3p91.a4.game.GameEngine;
|
||||||
import ca.cosc3p91.a4.game.Map;
|
import ca.cosc3p91.a4.game.Map;
|
||||||
|
import ca.cosc3p91.a4.gameobjects.Infantry;
|
||||||
|
import ca.cosc3p91.a4.gameobjects.Inhabitant;
|
||||||
|
|
||||||
import java.io.*;
|
import java.io.*;
|
||||||
import java.net.*;
|
import java.net.*;
|
||||||
import java.nio.ByteBuffer;
|
|
||||||
import java.nio.charset.StandardCharsets;
|
|
||||||
import java.rmi.ServerException;
|
import java.rmi.ServerException;
|
||||||
|
|
||||||
import java.util.ArrayList;
|
import java.util.*;
|
||||||
import java.util.HashMap;
|
import java.util.concurrent.LinkedBlockingQueue;
|
||||||
import java.util.PriorityQueue;
|
|
||||||
import java.util.Queue;
|
|
||||||
import java.util.concurrent.atomic.AtomicBoolean;
|
import java.util.concurrent.atomic.AtomicBoolean;
|
||||||
import java.util.concurrent.locks.ReentrantLock;
|
import java.util.concurrent.locks.ReentrantLock;
|
||||||
|
|
||||||
|
@ -26,9 +24,9 @@ public class Server implements Runnable {
|
||||||
private long clientAssignmentID = 0;
|
private long clientAssignmentID = 0;
|
||||||
private final DatagramSocket socket;
|
private final DatagramSocket socket;
|
||||||
private final Thread ioThread;
|
private final Thread ioThread;
|
||||||
private long lastSentMessageID = 0;
|
// private static volatile long lastSentMessageID = 0;
|
||||||
|
|
||||||
private GameEngine mainEngine;
|
private final GameEngine mainEngine;
|
||||||
|
|
||||||
private volatile boolean running = true;
|
private volatile boolean running = true;
|
||||||
|
|
||||||
|
@ -55,20 +53,20 @@ public class Server implements Runnable {
|
||||||
long clientID = stream.readLong();
|
long clientID = stream.readLong();
|
||||||
long messageID = stream.readLong();
|
long messageID = stream.readLong();
|
||||||
|
|
||||||
System.out.println("Receiving message with ID " + messageID + " to client: " + clientID + " of type " + packetID);
|
System.out.println("Receiving message with ID " + messageID + " from client: " + clientID + " of type " + packetID);
|
||||||
|
|
||||||
ConnectedClient client = clients.get(clientID);
|
ConnectedClient client = clients.get(clientID);
|
||||||
|
|
||||||
// the server must handle connection requests while the client's processing thread will handle all other messages
|
// the server must handle connection requests while the client's processing thread will handle all other messages
|
||||||
if (packetID == PacketTable.CONNECT){
|
if (packetID == PacketIDs.CONNECT){
|
||||||
long cid = ++clientAssignmentID;
|
long cid = ++clientAssignmentID;
|
||||||
System.out.println("A client has connected, his clientID is " + cid);
|
System.out.println("A client has connected, his clientID is " + cid);
|
||||||
clients.put(cid, new ConnectedClient(socket, mainEngine, cid, messageID, receivePacket.getAddress(), receivePacket.getPort()));
|
clients.put(cid, new ConnectedClient(this, mainEngine, cid, messageID, receivePacket.getAddress(), receivePacket.getPort()));
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
if (client == null)
|
if (client == null)
|
||||||
throw new ServerException("Client disconnected with invalid client id! (" + clientID + ")");
|
throw new ServerException("Client sent message invalid client id! (" + clientID + ")");
|
||||||
if (packetID == PacketTable.DISCONNECT) {
|
if (packetID == PacketIDs.DISCONNECT) {
|
||||||
client.halt();
|
client.halt();
|
||||||
clients.remove(clientID);
|
clients.remove(clientID);
|
||||||
continue;
|
continue;
|
||||||
|
@ -85,27 +83,25 @@ public class Server implements Runnable {
|
||||||
ioThread.join();
|
ioThread.join();
|
||||||
}
|
}
|
||||||
|
|
||||||
public static void main(String[] args) throws IOException, InterruptedException {
|
public static class ConnectedClient implements Runnable {
|
||||||
new Server();
|
|
||||||
}
|
|
||||||
|
|
||||||
private static class ConnectedClient implements Runnable {
|
|
||||||
private final InetAddress address;
|
private final InetAddress address;
|
||||||
private final int port;
|
private final int port;
|
||||||
private final Queue<Message.Received> pendingRequests = new PriorityQueue<>();
|
private final Queue<Message.Received> pendingRequests = new LinkedBlockingQueue<>();
|
||||||
private final ReentrantLock requestLock = new ReentrantLock();
|
private final ReentrantLock requestLock = new ReentrantLock();
|
||||||
private final AtomicBoolean allowUpdate;
|
private final AtomicBoolean allowUpdate;
|
||||||
private final HashMap<Long, Message.Sent> sentMessages = new HashMap<>();
|
private final java.util.Map<Long, Message.Sent> sentMessages = Collections.synchronizedMap(new HashMap<>());
|
||||||
private final DatagramSocket serverSocket;
|
private final Server server;
|
||||||
private final long clientID;
|
private final long clientID;
|
||||||
private volatile boolean running = true;
|
private volatile boolean running = true;
|
||||||
private final Thread processingThread;
|
private final Thread processingThread;
|
||||||
private final Thread gameEngineThread;
|
private final Thread gameEngineThread;
|
||||||
private final GameEngine usingEngine;
|
private final GameEngine usingEngine;
|
||||||
|
private long lastSentMessageID = 0;
|
||||||
private final Map clientMap;
|
private final Map clientMap;
|
||||||
|
private Map exploringMap;
|
||||||
|
|
||||||
public ConnectedClient(DatagramSocket serverSocket, GameEngine engine, long clientID, long messageID, InetAddress address, int port){
|
public ConnectedClient(Server server, GameEngine engine, long clientID, long messageID, InetAddress address, int port){
|
||||||
this.serverSocket = serverSocket;
|
this.server = server;
|
||||||
this.address = address;
|
this.address = address;
|
||||||
this.port = port;
|
this.port = port;
|
||||||
this.clientID = clientID;
|
this.clientID = clientID;
|
||||||
|
@ -123,7 +119,7 @@ public class Server implements Runnable {
|
||||||
});
|
});
|
||||||
gameEngineThread.start();
|
gameEngineThread.start();
|
||||||
|
|
||||||
sendMessage(new Message.Sent(PacketTable.ACK, clientID, messageID));
|
sendMessage(new Message.Sent(PacketIDs.ACK, clientID, messageID));
|
||||||
}
|
}
|
||||||
|
|
||||||
public void handleRequest(Message.Received request){
|
public void handleRequest(Message.Received request){
|
||||||
|
@ -137,27 +133,121 @@ public class Server implements Runnable {
|
||||||
private void processRequest(Message.Received request){
|
private void processRequest(Message.Received request){
|
||||||
try {
|
try {
|
||||||
switch (request.getPacketID()) {
|
switch (request.getPacketID()) {
|
||||||
case PacketTable.ACK:
|
case PacketIDs.ACK:
|
||||||
Message.Sent message = sentMessages.get(request.getMessageID());
|
Message.Sent message = sentMessages.get(request.getMessageID());
|
||||||
if (message == null)
|
if (message == null)
|
||||||
throw new RuntimeException("A message was acknowledged but does not exist!");
|
throw new RuntimeException("A message was acknowledged but does not exist!");
|
||||||
message.acknowledged();
|
message.acknowledged();
|
||||||
sentMessages.remove(request.getMessageID());
|
sentMessages.remove(request.getMessageID());
|
||||||
|
synchronized (sentMessages) {
|
||||||
|
sentMessages.notifyAll();
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
case PacketTable.MESSAGE:
|
case PacketIDs.MESSAGE:
|
||||||
System.out.println(request.getReader().readUTF());
|
System.out.println(request.getReader().readUTF());
|
||||||
break;
|
break;
|
||||||
case PacketTable.BUILD:
|
case PacketIDs.BUILD:
|
||||||
usingEngine.build(clientMap,new String(request.getData(), StandardCharsets.UTF_8));
|
try {
|
||||||
break;
|
String type = request.getReader().readUTF().trim();
|
||||||
case PacketTable.TRAIN:
|
if (usingEngine.build(clientMap, type))
|
||||||
usingEngine.train(clientMap,new String(request.getData(), StandardCharsets.UTF_8));
|
sendAndLogLn("Client " + clientID + " has successfully built " + type + "!");
|
||||||
break;
|
else
|
||||||
case PacketTable.UPGRADE:
|
sendAndLogLn("Client " + clientID + " has insufficient funds to build " + type + "!");
|
||||||
usingEngine.upgradeBuilding(clientMap, ByteBuffer.wrap(request.getData()).getInt());
|
} catch (GameEngine.BuildingErrorException e){
|
||||||
break;
|
sendAndLogLn(e.getMessage());
|
||||||
|
|
||||||
}
|
}
|
||||||
|
break;
|
||||||
|
case PacketIDs.TRAIN:
|
||||||
|
try {
|
||||||
|
String type = request.getReader().readUTF().trim();
|
||||||
|
if (usingEngine.train(clientMap, type))
|
||||||
|
sendAndLogLn("Client " + clientID + " has successfully trained " + type + "!");
|
||||||
|
else
|
||||||
|
sendAndLogLn("Client " + clientID + " has insufficient funds to train " + type + "!");
|
||||||
|
} catch (GameEngine.TrainingErrorException e){
|
||||||
|
sendAndLogLn(e.getMessage());
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case PacketIDs.UPGRADE:
|
||||||
|
try {
|
||||||
|
String type = request.getReader().readUTF();
|
||||||
|
int val = Integer.parseInt(
|
||||||
|
type.replace("b", "")
|
||||||
|
.replace(" ", "")
|
||||||
|
.replace("i", "")
|
||||||
|
.trim());
|
||||||
|
boolean status = false;
|
||||||
|
if (type.contains("b"))
|
||||||
|
status = usingEngine.upgradeBuilding(clientMap, val);
|
||||||
|
else
|
||||||
|
status = usingEngine.upgradeInhabitant(clientMap, val);
|
||||||
|
if (status)
|
||||||
|
sendAndLogLn("Client " + clientID + " has successfully upgraded " + type + "!");
|
||||||
|
else
|
||||||
|
sendAndLogLn("Client " + clientID + " was unable to upgrade " + type + "!");
|
||||||
|
} catch (GameEngine.UpgradingErrorException e){
|
||||||
|
sendAndLogLn(e.getMessage());
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case PacketIDs.PRINT_MAP_DATA:
|
||||||
|
sendMapData(usingEngine.view.getVillageStateTable(clientMap, "Home Village"));
|
||||||
|
break;
|
||||||
|
case PacketIDs.EXPLORE:
|
||||||
|
Random rand = new Random();
|
||||||
|
int clients = server.clients.size();
|
||||||
|
if (clients <= 1) {
|
||||||
|
sendAndLogLn("No other clients are currently connected! Please generate a village!");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
int pos = rand.nextInt(clients);
|
||||||
|
long findingClientID = clientID;
|
||||||
|
ConnectedClient foundClient = null;
|
||||||
|
int searchCount = 0;
|
||||||
|
while (findingClientID == clientID) {
|
||||||
|
pos = rand.nextInt(clients);
|
||||||
|
Iterator<java.util.Map.Entry<Long, ConnectedClient>> entries = server.clients.entrySet().iterator();
|
||||||
|
for (int i = 0; i < pos; i++)
|
||||||
|
entries.next();
|
||||||
|
foundClient = entries.next().getValue();
|
||||||
|
findingClientID = foundClient.clientID;
|
||||||
|
if(searchCount++ > 50)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
exploringMap = foundClient.clientMap;
|
||||||
|
sendMapData(usingEngine.view.getVillageStateTable(exploringMap, "Other Village"));
|
||||||
|
break;
|
||||||
|
case PacketIDs.GENERATE:
|
||||||
|
exploringMap = usingEngine.generateMap(clientMap);
|
||||||
|
sendMapData(usingEngine.view.getVillageStateTable(exploringMap, "Generated Village"));
|
||||||
|
break;
|
||||||
|
case PacketIDs.ATTACK:
|
||||||
|
if (exploringMap != null) {
|
||||||
|
if (!usingEngine.attackVillage(clientMap, exploringMap, false,this))
|
||||||
|
sendAndLogLn("Failed to attack!");
|
||||||
|
} else
|
||||||
|
sendAndLogLn("Error: Explored map is null. Did you explored/generated last command?");
|
||||||
|
exploringMap = null;
|
||||||
|
break;
|
||||||
|
case PacketIDs.TEST_ARMY:
|
||||||
|
case PacketIDs.TEST_VILLAGE:
|
||||||
|
// said it had to be similar, not that it couldn't be the same!
|
||||||
|
Map m = usingEngine.generateInitialMap();
|
||||||
|
clientMap.inhabitants.stream().filter(i -> i instanceof Infantry).forEach(i -> {
|
||||||
|
m.inhabitants.add(i);
|
||||||
|
});
|
||||||
|
Random iamtired = new Random(69420);
|
||||||
|
int goodnight = iamtired.nextInt(69420 * 2 + 1) - 69420;
|
||||||
|
String asillynightmare = request.getPacketID() == PacketIDs.TEST_VILLAGE
|
||||||
|
? ("Your score was: " + String.valueOf(goodnight) + (goodnight <= 0 ? "! (you suck)" : "!"))
|
||||||
|
: "";
|
||||||
|
if (usingEngine.attackVillage(m, clientMap, true, this))
|
||||||
|
sendAndLogLn("Your village failed to defend! " + asillynightmare);
|
||||||
|
else
|
||||||
|
sendAndLogLn("Your village successfully defended itself from a similarly sized army! " + asillynightmare);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (request.getPacketID() != PacketIDs.ACK)
|
||||||
|
sendMessage(new Message.Sent(PacketIDs.ACK, clientID, request.getMessageID()));
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
throw new RuntimeException(e);
|
throw new RuntimeException(e);
|
||||||
}
|
}
|
||||||
|
@ -175,8 +265,11 @@ public class Server implements Runnable {
|
||||||
}
|
}
|
||||||
requestLock.unlock();
|
requestLock.unlock();
|
||||||
|
|
||||||
|
// sentEntries needn't be in the synchronized block
|
||||||
|
Set<HashMap.Entry<Long, Message.Sent>> sentEntries = sentMessages.entrySet();
|
||||||
|
synchronized (sentMessages) {
|
||||||
ArrayList<Long> removes = new ArrayList<>();
|
ArrayList<Long> removes = new ArrayList<>();
|
||||||
for (HashMap.Entry<Long, Message.Sent> message : sentMessages.entrySet()){
|
for (HashMap.Entry<Long, Message.Sent> message : sentEntries) {
|
||||||
Message.Sent sent = message.getValue();
|
Message.Sent sent = message.getValue();
|
||||||
if (!sent.isAcknowledged() && sent.getTimeSinceSent().get() > MAX_PACKET_ACK_TIME_SECONDS) {
|
if (!sent.isAcknowledged() && sent.getTimeSinceSent().get() > MAX_PACKET_ACK_TIME_SECONDS) {
|
||||||
System.out.println("The client did not acknowledge our message, did they receive it?");
|
System.out.println("The client did not acknowledge our message, did they receive it?");
|
||||||
|
@ -188,9 +281,57 @@ public class Server implements Runnable {
|
||||||
sentMessages.remove(l);
|
sentMessages.remove(l);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void sendMapData(ArrayList<String> lines) {
|
||||||
|
final long messageID = ++lastSentMessageID;
|
||||||
|
Message.Sent beginMapInfoMessage = new Message.Sent(PacketIDs.BEGIN_MAP_DATA, clientID, messageID);
|
||||||
|
try {
|
||||||
|
beginMapInfoMessage.getWriter().writeInt(lines.size());
|
||||||
|
sendMessage(beginMapInfoMessage);
|
||||||
|
} catch (IOException e) {
|
||||||
|
sendAndLogLn("Unable to send map data: " + e.getMessage());
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
new Thread(() -> {
|
||||||
|
while (sentMessages.containsKey(messageID)){
|
||||||
|
try {
|
||||||
|
synchronized (sentMessages) {
|
||||||
|
sentMessages.wait();
|
||||||
|
}
|
||||||
|
} catch (InterruptedException e) {
|
||||||
|
e.printStackTrace();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// once we know that the client is waiting on our map data, we can send it in any order.
|
||||||
|
for (int i = 0; i < lines.size(); i++){
|
||||||
|
Message.Sent line = new Message.Sent(PacketIDs.MAP_LINE_DATA, clientID, ++lastSentMessageID);
|
||||||
|
try {
|
||||||
|
// but we need the line index!
|
||||||
|
line.getWriter().writeInt(i);
|
||||||
|
line.getWriter().writeUTF(lines.get(i));
|
||||||
|
} catch (IOException e){
|
||||||
|
e.printStackTrace();
|
||||||
|
}
|
||||||
|
sendMessage(line);
|
||||||
|
}
|
||||||
|
}).start();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void sendAndLogLn(String str){
|
||||||
|
Message.Sent mess = new Message.Sent(PacketIDs.MESSAGE, clientID, ++lastSentMessageID);
|
||||||
|
try {
|
||||||
|
mess.getWriter().writeUTF(str + "\n");
|
||||||
|
sendMessage(mess);
|
||||||
|
System.out.println(str);
|
||||||
|
} catch (IOException e){
|
||||||
|
e.printStackTrace();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
public void sendMessage(Message.Sent message){
|
public void sendMessage(Message.Sent message){
|
||||||
if (message.getPacketID() != PacketTable.ACK)
|
if (message.getPacketID() != PacketIDs.ACK)
|
||||||
this.sentMessages.put(message.getMessageID(), message);
|
this.sentMessages.put(message.getMessageID(), message);
|
||||||
byte[] data = message.getData().toByteArray();
|
byte[] data = message.getData().toByteArray();
|
||||||
if (data.length > PACKET_SIZE)
|
if (data.length > PACKET_SIZE)
|
||||||
|
@ -198,7 +339,7 @@ public class Server implements Runnable {
|
||||||
DatagramPacket request = new DatagramPacket(data, data.length, address, port);
|
DatagramPacket request = new DatagramPacket(data, data.length, address, port);
|
||||||
try {
|
try {
|
||||||
System.out.println("Sending message with ID " + message.getMessageID() + " to client: " + message.getClientID() + " of type " + message.getPacketID());
|
System.out.println("Sending message with ID " + message.getMessageID() + " to client: " + message.getClientID() + " of type " + message.getPacketID());
|
||||||
serverSocket.send(request);
|
server.socket.send(request);
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
e.printStackTrace();
|
e.printStackTrace();
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue