This commit is contained in:
Anuken
2019-08-19 22:36:28 -04:00
14 changed files with 143 additions and 202 deletions

View File

@@ -124,7 +124,7 @@ save.new = New Save
save.overwrite = Are you sure you want to overwrite\nthis save slot? save.overwrite = Are you sure you want to overwrite\nthis save slot?
overwrite = Overwrite overwrite = Overwrite
save.none = No saves found! save.none = No saves found!
saveload = [accent]Saving... saveload = Saving...
savefail = Failed to save game! savefail = Failed to save game!
save.delete.confirm = Are you sure you want to delete this save? save.delete.confirm = Are you sure you want to delete this save?
save.delete = Delete save.delete = Delete

View File

@@ -119,7 +119,7 @@ public class BlockIndexer{
ObjectSet<Tile> set = damagedTiles[team.ordinal()]; ObjectSet<Tile> set = damagedTiles[team.ordinal()];
for(Tile tile : set){ for(Tile tile : set){
if((tile.entity == null || tile.entity.getTeam() != team || !tile.entity.damaged()) && !(tile.block() instanceof BuildBlock)){ if((tile.entity == null || tile.entity.getTeam() != team || !tile.entity.damaged()) || tile.block() instanceof BuildBlock){
returnArray.add(tile); returnArray.add(tile);
} }
} }

View File

@@ -84,7 +84,7 @@ public class Control implements ApplicationListener{
}); });
Events.on(PlayEvent.class, event -> { Events.on(PlayEvent.class, event -> {
player.setTeam(defaultTeam); player.setTeam(state.rules.pvp ? netServer.assignTeam(playerGroup.all()) : defaultTeam);
player.setDead(true); player.setDead(true);
player.add(); player.add();
@@ -237,7 +237,7 @@ public class Control implements ApplicationListener{
public void playMap(Map map, Rules rules){ public void playMap(Map map, Rules rules){
ui.loadAnd(() -> { ui.loadAnd(() -> {
logic.reset(); logic.reset();
world.loadMap(map); world.loadMap(map, rules);
state.rules = rules; state.rules = rules;
logic.play(); logic.play();
}); });

View File

@@ -40,7 +40,7 @@ import static io.anuke.mindustry.Vars.*;
public class NetServer implements ApplicationListener{ public class NetServer implements ApplicationListener{
public final static int maxSnapshotSize = 430; public final static int maxSnapshotSize = 430;
private final static float serverSyncTime = 15, kickDuration = 30 * 1000; private final static float serverSyncTime = 12, kickDuration = 30 * 1000;
private final static Vector2 vector = new Vector2(); private final static Vector2 vector = new Vector2();
private final static Rectangle viewport = new Rectangle(); private final static Rectangle viewport = new Rectangle();
/** If a player goes away of their server-side coordinates by this distance, they get teleported back. */ /** If a player goes away of their server-side coordinates by this distance, they get teleported back. */

View File

@@ -220,6 +220,10 @@ public class World implements ApplicationListener{
} }
public void loadMap(Map map){ public void loadMap(Map map){
loadMap(map, new Rules());
}
public void loadMap(Map map, Rules checkRules){
try{ try{
SaveIO.load(map.file, new FilterContext(map)); SaveIO.load(map.file, new FilterContext(map));
}catch(Exception e){ }catch(Exception e){
@@ -238,20 +242,21 @@ public class World implements ApplicationListener{
invalidMap = false; invalidMap = false;
if(!headless){ if(!headless){
if(state.teams.get(defaultTeam).cores.size == 0){ if(state.teams.get(defaultTeam).cores.size == 0 && !checkRules.pvp){
ui.showError("$map.nospawn"); ui.showError("$map.nospawn");
invalidMap = true; invalidMap = true;
}else if(state.rules.pvp){ //pvp maps need two cores to be valid }else if(checkRules.pvp){ //pvp maps need two cores to be valid
invalidMap = true; int teams = 0;
for(Team team : Team.all){ for(Team team : Team.all){
if(state.teams.get(team).cores.size != 0 && team != defaultTeam){ if(state.teams.get(team).cores.size != 0){
invalidMap = false; teams ++;
} }
} }
if(invalidMap){ if(teams < 2){
invalidMap = true;
ui.showError("$map.nospawn.pvp"); ui.showError("$map.nospawn.pvp");
} }
}else if(state.rules.attackMode){ //pvp maps need two cores to be valid }else if(checkRules.attackMode){ //attack maps need two cores to be valid
invalidMap = state.teams.get(waveTeam).cores.isEmpty(); invalidMap = state.teams.get(waveTeam).cores.isEmpty();
if(invalidMap){ if(invalidMap){
ui.showError("$map.nospawn.attack"); ui.showError("$map.nospawn.attack");

View File

@@ -1,33 +1,28 @@
package io.anuke.mindustry.entities.traits; package io.anuke.mindustry.entities.traits;
import io.anuke.annotations.Annotations.*; import io.anuke.annotations.Annotations.*;
import io.anuke.arc.Core; import io.anuke.arc.*;
import io.anuke.arc.Events;
import io.anuke.arc.collection.Array;
import io.anuke.arc.collection.Queue; import io.anuke.arc.collection.Queue;
import io.anuke.arc.collection.*;
import io.anuke.arc.graphics.g2d.*; import io.anuke.arc.graphics.g2d.*;
import io.anuke.arc.math.Angles; import io.anuke.arc.math.*;
import io.anuke.arc.math.Mathf; import io.anuke.arc.math.geom.*;
import io.anuke.arc.math.geom.Vector2; import io.anuke.arc.util.*;
import io.anuke.arc.util.Time; import io.anuke.mindustry.*;
import io.anuke.mindustry.Vars; import io.anuke.mindustry.content.*;
import io.anuke.mindustry.content.Blocks; import io.anuke.mindustry.entities.type.*;
import io.anuke.mindustry.entities.type.TileEntity; import io.anuke.mindustry.game.EventType.*;
import io.anuke.mindustry.entities.type.Unit; import io.anuke.mindustry.gen.*;
import io.anuke.mindustry.game.EventType.BuildSelectEvent; import io.anuke.mindustry.graphics.*;
import io.anuke.mindustry.gen.Call;
import io.anuke.mindustry.graphics.Pal;
import io.anuke.mindustry.net.Net;
import io.anuke.mindustry.world.*; import io.anuke.mindustry.world.*;
import io.anuke.mindustry.world.blocks.BuildBlock; import io.anuke.mindustry.world.blocks.*;
import io.anuke.mindustry.world.blocks.BuildBlock.BuildEntity; import io.anuke.mindustry.world.blocks.BuildBlock.*;
import java.io.*; import java.io.*;
import java.util.Arrays; import java.util.*;
import static io.anuke.mindustry.Vars.*; import static io.anuke.mindustry.Vars.*;
import static io.anuke.mindustry.entities.traits.BuilderTrait.BuildDataStatic.removal; import static io.anuke.mindustry.entities.traits.BuilderTrait.BuildDataStatic.*;
import static io.anuke.mindustry.entities.traits.BuilderTrait.BuildDataStatic.tmptr;
/** Interface for units that build things.*/ /** Interface for units that build things.*/
public interface BuilderTrait extends Entity, TeamTrait{ public interface BuilderTrait extends Entity, TeamTrait{
@@ -106,19 +101,14 @@ public interface BuilderTrait extends Entity, TeamTrait{
unit.rotation = Mathf.slerpDelta(unit.rotation, unit.angleTo(entity), 0.4f); unit.rotation = Mathf.slerpDelta(unit.rotation, unit.angleTo(entity), 0.4f);
} }
//progress is synced, thus not updated clientside //deconstructing is 2x as fast
if(!Net.client()){ if(current.breaking){
//deconstructing is 2x as fast entity.deconstruct(unit, core, 2f / entity.buildCost * Time.delta() * getBuildPower(tile) * state.rules.buildSpeedMultiplier);
if(current.breaking){
entity.deconstruct(unit, core, 2f / entity.buildCost * Time.delta() * getBuildPower(tile) * state.rules.buildSpeedMultiplier);
}else{
entity.construct(unit, core, 1f / entity.buildCost * Time.delta() * getBuildPower(tile) * state.rules.buildSpeedMultiplier);
}
current.progress = entity.progress();
}else{ }else{
entity.progress = current.progress; entity.construct(unit, core, 1f / entity.buildCost * Time.delta() * getBuildPower(tile) * state.rules.buildSpeedMultiplier);
} }
current.progress = entity.progress;
} }
/** Returns the queue for storing build requests. */ /** Returns the queue for storing build requests. */
@@ -135,7 +125,7 @@ public interface BuilderTrait extends Entity, TeamTrait{
default void writeBuilding(DataOutput output) throws IOException{ default void writeBuilding(DataOutput output) throws IOException{
BuildRequest request = buildRequest(); BuildRequest request = buildRequest();
if(request != null){ if(request != null && (request.block != null || request.breaking)){
output.writeByte(request.breaking ? 1 : 0); output.writeByte(request.breaking ? 1 : 0);
output.writeInt(Pos.get(request.x, request.y)); output.writeInt(Pos.get(request.x, request.y));
output.writeFloat(request.progress); output.writeFloat(request.progress);
@@ -174,7 +164,11 @@ public interface BuilderTrait extends Entity, TeamTrait{
if(applyChanges){ if(applyChanges){
buildQueue().addLast(request); buildQueue().addLast(request);
}else if(isBuilding()){ }else if(isBuilding()){
buildRequest().progress = progress; BuildRequest last = buildRequest();
last.progress = progress;
if(last.tile() != null && last.tile().entity instanceof BuildEntity){
((BuildEntity)last.tile().entity).progress = progress;
}
} }
} }
} }

View File

@@ -5,6 +5,7 @@ import io.anuke.mindustry.entities.type.TileEntity;
import io.anuke.mindustry.entities.units.UnitState; import io.anuke.mindustry.entities.units.UnitState;
import io.anuke.mindustry.world.Pos; import io.anuke.mindustry.world.Pos;
import io.anuke.mindustry.world.Tile; import io.anuke.mindustry.world.Tile;
import io.anuke.mindustry.world.blocks.*;
import java.io.*; import java.io.*;
@@ -23,6 +24,10 @@ public class RepairDrone extends BaseDrone{
target = Units.findDamagedTile(team, x, y); target = Units.findDamagedTile(team, x, y);
} }
if(target instanceof TileEntity && ((TileEntity)target).block instanceof BuildBlock){
target = null;
}
if(target != null){ if(target != null){
if(target.dst(RepairDrone.this) > type.range){ if(target.dst(RepairDrone.this) > type.range){
circle(type.range * 0.9f); circle(type.range * 0.9f);

View File

@@ -1,14 +1,15 @@
package io.anuke.mindustry.net; package io.anuke.mindustry.net;
import io.anuke.arc.math.Mathf; import io.anuke.arc.math.*;
import io.anuke.arc.math.geom.Vector2; import io.anuke.arc.math.geom.*;
import io.anuke.arc.util.Time; import io.anuke.arc.util.*;
public class Interpolator{ public class Interpolator{
//used for movement //used for movement
public Vector2 target = new Vector2(); public Vector2 target = new Vector2();
public Vector2 last = new Vector2(); public Vector2 last = new Vector2();
public float[] targets = {}; public float[] targets = {};
public float[] lasts = {};
public long lastUpdated, updateSpacing; public long lastUpdated, updateSpacing;
//current state //current state
@@ -21,6 +22,12 @@ public class Interpolator{
lastUpdated = Time.millis(); lastUpdated = Time.millis();
targets = target1ds; targets = target1ds;
if(lasts.length != values.length){
lasts = new float[values.length];
}
for(int i = 0; i < values.length; i++){
lasts[i] = values[i];
}
last.set(cx, cy); last.set(cx, cy);
target.set(x, y); target.set(x, y);
} }
@@ -46,8 +53,12 @@ public class Interpolator{
values = new float[targets.length]; values = new float[targets.length];
} }
if(lasts.length != targets.length){
lasts = new float[targets.length];
}
for(int i = 0; i < values.length; i++){ for(int i = 0; i < values.length; i++){
values[i] = Mathf.slerp(values[i], targets[i], alpha); values[i] = Mathf.slerp(lasts[i], targets[i], alpha);
} }
}else{ }else{
pos.set(target); pos.set(target);

View File

@@ -162,8 +162,8 @@ public class Net{
/** /**
* Returns a list of all connections IDs. * Returns a list of all connections IDs.
*/ */
public static Array<NetConnection> getConnections(){ public static Iterable<NetConnection> getConnections(){
return (Array<NetConnection>)serverProvider.getConnections(); return (Iterable<NetConnection>)serverProvider.getConnections();
} }
/** /**
@@ -377,16 +377,53 @@ public class Net{
void host(int port) throws IOException; void host(int port) throws IOException;
/** Sends a large stream of data to a specific client. */ /** Sends a large stream of data to a specific client. */
void sendStream(int id, Streamable stream); default void sendStream(int id, Streamable stream){
NetConnection connection = getByID(id);
if(connection == null) return;
try{
int cid;
StreamBegin begin = new StreamBegin();
begin.total = stream.stream.available();
begin.type = Registrator.getID(stream.getClass());
connection.send(begin, SendMode.tcp);
cid = begin.id;
/** Send an object to everyone connected. */ while(stream.stream.available() > 0){
void send(Object object, SendMode mode); byte[] bytes = new byte[Math.min(512, stream.stream.available())];
stream.stream.read(bytes);
/** Send an object to a specific client ID. */ StreamChunk chunk = new StreamChunk();
void sendTo(int id, Object object, SendMode mode); chunk.id = cid;
chunk.data = bytes;
connection.send(chunk, SendMode.tcp);
}
}catch(IOException e){
throw new RuntimeException(e);
}
}
/** Send an object to everyone <i>except</i> a client ID. */ default void send(Object object, SendMode mode){
void sendExcept(int id, Object object, SendMode mode); for(NetConnection con : getConnections()){
con.send(object, mode);
}
}
default void sendTo(int id, Object object, SendMode mode){
NetConnection conn = getByID(id);
if(conn == null){
Log.err("Failed to find connection with ID {0}.", id);
return;
}
conn.send(object, mode);
}
default void sendExcept(int id, Object object, SendMode mode){
for(NetConnection con : getConnections()){
if(con.id != id){
con.send(object, mode);
}
}
}
/** Close the server connection. */ /** Close the server connection. */
void close(); void close();
@@ -395,7 +432,7 @@ public class Net{
byte[] compressSnapshot(byte[] input); byte[] compressSnapshot(byte[] input);
/** Return all connected users. */ /** Return all connected users. */
Array<? extends NetConnection> getConnections(); Iterable<? extends NetConnection> getConnections();
/** Returns a connection by ID. */ /** Returns a connection by ID. */
NetConnection getByID(int id); NetConnection getByID(int id);

View File

@@ -3,6 +3,8 @@ package io.anuke.mindustry.net;
import io.anuke.mindustry.net.Net.SendMode; import io.anuke.mindustry.net.Net.SendMode;
public abstract class NetConnection{ public abstract class NetConnection{
private static int lastID;
public final int id; public final int id;
public final String address; public final String address;
@@ -18,8 +20,9 @@ public abstract class NetConnection{
public boolean hasBegunConnecting = false; public boolean hasBegunConnecting = false;
public float viewWidth, viewHeight, viewX, viewY; public float viewWidth, viewHeight, viewX, viewY;
public NetConnection(int id, String address){ /** Assigns this connection a unique ID. No two connections will ever have the same ID.*/
this.id = id; public NetConnection(String address){
this.id = lastID++;
this.address = address; this.address = address;
} }

View File

@@ -106,7 +106,7 @@ public class PausedDialog extends FloatingDialog{
return; return;
} }
if(control.saves.getCurrent() == null || !control.saves.getCurrent().isAutosave() || state.rules.tutorial){ if(control.saves.getCurrent() == null || !control.saves.getCurrent().isAutosave() || state.rules.tutorial || Net.client()){
state.set(State.menu); state.set(State.menu);
logic.reset(); logic.reset();
return; return;

View File

@@ -4,6 +4,7 @@ import io.anuke.arc.*;
import io.anuke.arc.collection.*; import io.anuke.arc.collection.*;
import io.anuke.arc.function.*; import io.anuke.arc.function.*;
import io.anuke.arc.net.*; import io.anuke.arc.net.*;
import io.anuke.arc.util.async.*;
import io.anuke.arc.util.pooling.*; import io.anuke.arc.util.pooling.*;
import io.anuke.mindustry.net.Net.*; import io.anuke.mindustry.net.Net.*;
import io.anuke.mindustry.net.Packets.*; import io.anuke.mindustry.net.Packets.*;
@@ -84,7 +85,7 @@ public class ArcNetClient implements ClientProvider{
@Override @Override
public void connect(String ip, int port, Runnable success){ public void connect(String ip, int port, Runnable success){
runAsync(() -> { Threads.daemon(() -> {
try{ try{
//just in case //just in case
client.stop(); client.stop();
@@ -99,7 +100,7 @@ public class ArcNetClient implements ClientProvider{
updateThread.setDaemon(true); updateThread.setDaemon(true);
updateThread.start(); updateThread.start();
client.connect(5000, ip, port, port); client.connect(5000, ip, port);
success.run(); success.run();
}catch(Exception e){ }catch(Exception e){
handleException(e); handleException(e);
@@ -115,11 +116,7 @@ public class ArcNetClient implements ClientProvider{
@Override @Override
public void send(Object object, SendMode mode){ public void send(Object object, SendMode mode){
try{ try{
if(mode == SendMode.tcp){ client.sendTCP(object);
client.sendTCP(object);
}else{
client.sendUDP(object);
}
//sending things can cause an under/overflow, catch it and disconnect instead of crashing //sending things can cause an under/overflow, catch it and disconnect instead of crashing
}catch(BufferOverflowException | BufferUnderflowException e){ }catch(BufferOverflowException | BufferUnderflowException e){
Net.showError(e); Net.showError(e);
@@ -140,7 +137,7 @@ public class ArcNetClient implements ClientProvider{
@Override @Override
public void pingHost(String address, int port, Consumer<Host> valid, Consumer<Exception> invalid){ public void pingHost(String address, int port, Consumer<Host> valid, Consumer<Exception> invalid){
runAsync(() -> { Threads.daemon(() -> {
try{ try{
DatagramSocket socket = new DatagramSocket(); DatagramSocket socket = new DatagramSocket();
socket.send(new DatagramPacket(new byte[]{-2, 1}, 2, InetAddress.getByName(address), port)); socket.send(new DatagramPacket(new byte[]{-2, 1}, 2, InetAddress.getByName(address), port));
@@ -189,12 +186,6 @@ public class ArcNetClient implements ClientProvider{
} }
} }
private void runAsync(Runnable run){
Thread thread = new Thread(run, "Client Async Run");
thread.setDaemon(true);
thread.start();
}
private void handleException(Exception e){ private void handleException(Exception e){
if(e instanceof ArcNetException){ if(e instanceof ArcNetException){
Core.app.post(() -> Net.showError(new IOException("mismatch"))); Core.app.post(() -> Net.showError(new IOException("mismatch")));

View File

@@ -1,9 +1,9 @@
package io.anuke.mindustry.net; package io.anuke.mindustry.net;
import io.anuke.arc.*; import io.anuke.arc.*;
import io.anuke.arc.collection.*;
import io.anuke.arc.net.*; import io.anuke.arc.net.*;
import io.anuke.arc.util.*; import io.anuke.arc.util.*;
import io.anuke.arc.util.async.*;
import io.anuke.mindustry.net.Net.*; import io.anuke.mindustry.net.Net.*;
import io.anuke.mindustry.net.Packets.*; import io.anuke.mindustry.net.Packets.*;
import net.jpountz.lz4.*; import net.jpountz.lz4.*;
@@ -17,14 +17,10 @@ import static io.anuke.mindustry.Vars.*;
public class ArcNetServer implements ServerProvider{ public class ArcNetServer implements ServerProvider{
final Server server; final Server server;
final CopyOnWriteArrayList<KryoConnection> connections = new CopyOnWriteArrayList<>(); final CopyOnWriteArrayList<ArcConnection> connections = new CopyOnWriteArrayList<>();
final CopyOnWriteArraySet<Integer> missing = new CopyOnWriteArraySet<>();
final Array<KryoConnection> array = new Array<>();
final LZ4Compressor compressor = LZ4Factory.fastestInstance().fastCompressor(); final LZ4Compressor compressor = LZ4Factory.fastestInstance().fastCompressor();
Thread serverThread; Thread serverThread;
int lastconnection = 0;
public ArcNetServer(){ public ArcNetServer(){
server = new Server(4096 * 2, 4096, new PacketSerializer()); server = new Server(4096 * 2, 4096, new PacketSerializer());
server.setMulticast(multicastGroup, multicastPort); server.setMulticast(multicastGroup, multicastPort);
@@ -40,7 +36,7 @@ public class ArcNetServer implements ServerProvider{
public void connected(Connection connection){ public void connected(Connection connection){
String ip = connection.getRemoteAddressTCP().getAddress().getHostAddress(); String ip = connection.getRemoteAddressTCP().getAddress().getHostAddress();
KryoConnection kn = new KryoConnection(lastconnection++, ip, connection); ArcConnection kn = new ArcConnection(ip, connection);
Connect c = new Connect(); Connect c = new Connect();
c.id = kn.id; c.id = kn.id;
@@ -54,7 +50,7 @@ public class ArcNetServer implements ServerProvider{
@Override @Override
public void disconnected(Connection connection){ public void disconnected(Connection connection){
KryoConnection k = getByKryoID(connection.getID()); ArcConnection k = getByKryoID(connection.getID());
if(k == null) return; if(k == null) return;
Disconnect c = new Disconnect(); Disconnect c = new Disconnect();
@@ -68,7 +64,7 @@ public class ArcNetServer implements ServerProvider{
@Override @Override
public void received(Connection connection, Object object){ public void received(Connection connection, Object object){
KryoConnection k = getByKryoID(connection.getID()); ArcConnection k = getByKryoID(connection.getID());
if(object instanceof FrameworkMessage || k == null) return; if(object instanceof FrameworkMessage || k == null) return;
Core.app.post(() -> { Core.app.post(() -> {
@@ -97,18 +93,14 @@ public class ArcNetServer implements ServerProvider{
} }
@Override @Override
public Array<KryoConnection> getConnections(){ public Iterable<ArcConnection> getConnections(){
array.clear(); return connections;
for(KryoConnection c : connections){
array.add(c);
}
return array;
} }
@Override @Override
public KryoConnection getByID(int id){ public ArcConnection getByID(int id){
for(int i = 0; i < connections.size(); i++){ for(int i = 0; i < connections.size(); i++){
KryoConnection con = connections.get(i); ArcConnection con = connections.get(i);
if(con.id == id){ if(con.id == id){
return con; return con;
} }
@@ -119,16 +111,14 @@ public class ArcNetServer implements ServerProvider{
@Override @Override
public void host(int port) throws IOException{ public void host(int port) throws IOException{
lastconnection = 0;
connections.clear(); connections.clear();
missing.clear(); server.bind(port);
server.bind(port, port);
serverThread = new Thread(() -> { serverThread = new Thread(() -> {
try{ try{
server.run(); server.run();
}catch(Throwable e){ }catch(Throwable e){
if(!(e instanceof ClosedSelectorException)) handleException(e); if(!(e instanceof ClosedSelectorException)) Threads.throwAppException(e);
} }
}, "Net Server"); }, "Net Server");
serverThread.setDaemon(true); serverThread.setDaemon(true);
@@ -138,97 +128,12 @@ public class ArcNetServer implements ServerProvider{
@Override @Override
public void close(){ public void close(){
connections.clear(); connections.clear();
lastconnection = 0; Threads.daemon(server::stop);
async(server::stop);
} }
@Override ArcConnection getByKryoID(int id){
public void sendStream(int id, Streamable stream){
KryoConnection connection = getByID(id);
if(connection == null) return;
try{
if(connection.connection != null){
connection.connection.addListener(new InputStreamSender(stream.stream, 512){
int id;
protected void start(){
//send an object so the receiving side knows how to handle the following chunks
StreamBegin begin = new StreamBegin();
begin.total = stream.stream.available();
begin.type = Registrator.getID(stream.getClass());
connection.connection.sendTCP(begin);
id = begin.id;
}
protected Object next(byte[] bytes){
StreamChunk chunk = new StreamChunk();
chunk.id = id;
chunk.data = bytes;
return chunk; //wrap the byte[] with an object so the receiving side knows how to handle it.
}
});
}else{
int cid;
StreamBegin begin = new StreamBegin();
begin.total = stream.stream.available();
begin.type = Registrator.getID(stream.getClass());
connection.send(begin, SendMode.tcp);
cid = begin.id;
while(stream.stream.available() > 0){
byte[] bytes = new byte[Math.min(512, stream.stream.available())];
stream.stream.read(bytes);
StreamChunk chunk = new StreamChunk();
chunk.id = cid;
chunk.data = bytes;
connection.send(chunk, SendMode.tcp);
}
}
}catch(IOException e){
throw new RuntimeException(e);
}
}
@Override
public void send(Object object, SendMode mode){
for(int i = 0; i < connections.size(); i++){ for(int i = 0; i < connections.size(); i++){
connections.get(i).send(object, mode); ArcConnection con = connections.get(i);
}
}
@Override
public void sendTo(int id, Object object, SendMode mode){
NetConnection conn = getByID(id);
if(conn == null){
if(!missing.contains(id))
Log.err("Failed to find connection with ID {0}.", id);
missing.add(id);
return;
}
conn.send(object, mode);
}
@Override
public void sendExcept(int id, Object object, SendMode mode){
for(int i = 0; i < connections.size(); i++){
KryoConnection conn = connections.get(i);
if(conn.id != id) conn.send(object, mode);
}
}
private void handleException(Throwable e){
Time.run(0f, () -> {
throw new RuntimeException(e);
});
}
KryoConnection getByKryoID(int id){
for(int i = 0; i < connections.size(); i++){
KryoConnection con = connections.get(i);
if(con.connection != null && con.connection.getID() == id){ if(con.connection != null && con.connection.getID() == id){
return con; return con;
} }
@@ -237,17 +142,11 @@ public class ArcNetServer implements ServerProvider{
return null; return null;
} }
void async(Runnable run){ class ArcConnection extends NetConnection{
Thread thread = new Thread(run);
thread.setDaemon(true);
thread.start();
}
class KryoConnection extends NetConnection{
public final Connection connection; public final Connection connection;
public KryoConnection(int id, String address, Connection connection){ public ArcConnection(String address, Connection connection){
super(id, address); super(address);
this.connection = connection; this.connection = connection;
} }
@@ -259,17 +158,13 @@ public class ArcNetServer implements ServerProvider{
@Override @Override
public void send(Object object, SendMode mode){ public void send(Object object, SendMode mode){
try{ try{
if(mode == SendMode.tcp){ connection.sendTCP(object);
connection.sendTCP(object);
}else{
connection.sendUDP(object);
}
}catch(Exception e){ }catch(Exception e){
Log.err(e); Log.err(e);
Log.info("Error sending packet. Disconnecting invalid client!"); Log.info("Error sending packet. Disconnecting invalid client!");
connection.close(); connection.close();
KryoConnection k = getByKryoID(connection.getID()); ArcConnection k = getByKryoID(connection.getID());
if(k != null) connections.remove(k); if(k != null) connections.remove(k);
} }
} }

View File

@@ -157,7 +157,7 @@ public class ServerControl implements ApplicationListener{
info("Selected next map to be {0}.", map.name()); info("Selected next map to be {0}.", map.name());
play(true, () -> world.loadMap(map)); play(true, () -> world.loadMap(map, map.applyRules(lastMode)));
} }
}else{ }else{
netServer.kickAll(KickReason.gameover); netServer.kickAll(KickReason.gameover);
@@ -231,7 +231,7 @@ public class ServerControl implements ApplicationListener{
logic.reset(); logic.reset();
lastMode = preset; lastMode = preset;
try{ try{
world.loadMap(result); world.loadMap(result, result.applyRules(lastMode));
state.rules = result.applyRules(preset); state.rules = result.applyRules(preset);
logic.play(); logic.play();