304 lines
9.5 KiB
Java
304 lines
9.5 KiB
Java
package io.anuke.kryonet;
|
|
|
|
import com.badlogic.gdx.Gdx;
|
|
import com.badlogic.gdx.utils.Array;
|
|
import com.esotericsoftware.kryonet.Connection;
|
|
import com.esotericsoftware.kryonet.FrameworkMessage;
|
|
import com.esotericsoftware.kryonet.Listener;
|
|
import com.esotericsoftware.kryonet.Listener.LagListener;
|
|
import com.esotericsoftware.kryonet.Server;
|
|
import com.esotericsoftware.kryonet.util.InputStreamSender;
|
|
import io.anuke.mindustry.net.Net;
|
|
import io.anuke.mindustry.net.Net.SendMode;
|
|
import io.anuke.mindustry.net.Net.ServerProvider;
|
|
import io.anuke.mindustry.net.NetConnection;
|
|
import io.anuke.mindustry.net.NetworkIO;
|
|
import io.anuke.mindustry.net.Packets.Connect;
|
|
import io.anuke.mindustry.net.Packets.Disconnect;
|
|
import io.anuke.mindustry.net.Packets.StreamBegin;
|
|
import io.anuke.mindustry.net.Packets.StreamChunk;
|
|
import io.anuke.mindustry.net.Streamable;
|
|
import io.anuke.ucore.UCore;
|
|
import io.anuke.ucore.core.Timers;
|
|
import io.anuke.ucore.util.Log;
|
|
|
|
import java.io.IOException;
|
|
import java.nio.ByteBuffer;
|
|
import java.nio.channels.ClosedSelectorException;
|
|
import java.util.concurrent.CopyOnWriteArrayList;
|
|
import java.util.concurrent.CopyOnWriteArraySet;
|
|
|
|
public class KryoServer implements ServerProvider {
|
|
final boolean tcpOnly = System.getProperty("java.version") == null;
|
|
final Server server;
|
|
final CopyOnWriteArrayList<KryoConnection> connections = new CopyOnWriteArrayList<>();
|
|
final CopyOnWriteArraySet<Integer> missing = new CopyOnWriteArraySet<>();
|
|
final Array<KryoConnection> array = new Array<>();
|
|
Thread serverThread;
|
|
|
|
int lastconnection = 0;
|
|
|
|
public KryoServer(){
|
|
KryoCore.init();
|
|
|
|
server = new Server(4096*2, 4096, connection -> new ByteSerializer());
|
|
server.setDiscoveryHandler((datagramChannel, fromAddress) -> {
|
|
ByteBuffer buffer = NetworkIO.writeServerData();
|
|
buffer.position(0);
|
|
datagramChannel.send(buffer, fromAddress);
|
|
return true;
|
|
});
|
|
|
|
Listener listener = new Listener(){
|
|
|
|
@Override
|
|
public void connected (Connection connection) {
|
|
String ip = connection.getRemoteAddressTCP().getAddress().getHostAddress();
|
|
|
|
KryoConnection kn = new KryoConnection(lastconnection ++, ip, connection);
|
|
|
|
Connect c = new Connect();
|
|
c.id = kn.id;
|
|
c.addressTCP = ip;
|
|
|
|
Log.info("&bRecieved connection: {0} / {1}. Kryonet ID: {2}", c.id, c.addressTCP, connection.getID());
|
|
|
|
connections.add(kn);
|
|
Gdx.app.postRunnable(() -> Net.handleServerReceived(kn.id, c));
|
|
}
|
|
|
|
@Override
|
|
public void disconnected (Connection connection) {
|
|
KryoConnection k = getByKryoID(connection.getID());
|
|
Log.info("&bLost kryonet connection {0}", connection.getID());
|
|
if(k == null) return;
|
|
|
|
Disconnect c = new Disconnect();
|
|
c.id = k.id;
|
|
|
|
Log.info("&bLost connection: {0}", k.id);
|
|
|
|
Gdx.app.postRunnable(() -> {
|
|
Net.handleServerReceived(k.id, c);
|
|
connections.remove(k);
|
|
});
|
|
}
|
|
|
|
@Override
|
|
public void received (Connection connection, Object object) {
|
|
KryoConnection k = getByKryoID(connection.getID());
|
|
if(object instanceof FrameworkMessage || k == null) return;
|
|
|
|
Gdx.app.postRunnable(() -> {
|
|
try{
|
|
Net.handleServerReceived(k.id, object);
|
|
}catch (Exception e){
|
|
e.printStackTrace();
|
|
}
|
|
});
|
|
}
|
|
};
|
|
|
|
if(KryoCore.fakeLag){
|
|
server.addListener(new LagListener(KryoCore.fakeLagMin, KryoCore.fakeLagMax, listener));
|
|
}else{
|
|
server.addListener(listener);
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public Array<KryoConnection> getConnections() {
|
|
array.clear();
|
|
for(KryoConnection c : connections){
|
|
array.add(c);
|
|
}
|
|
return array;
|
|
}
|
|
|
|
@Override
|
|
public KryoConnection getByID(int id){
|
|
for(int i = 0; i < connections.size(); i ++){
|
|
KryoConnection con = connections.get(i);
|
|
if(con.id == id){
|
|
return con;
|
|
}
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
@Override
|
|
public void host(int port) throws IOException {
|
|
lastconnection = 0;
|
|
connections.clear();
|
|
missing.clear();
|
|
if(tcpOnly){
|
|
server.bind(port);
|
|
}else{
|
|
server.bind(port, port);
|
|
}
|
|
|
|
serverThread = new Thread(() -> {
|
|
try{
|
|
server.run();
|
|
}catch (Throwable e){
|
|
if(!(e instanceof ClosedSelectorException)) handleException(e);
|
|
}
|
|
}, "Kryonet Server");
|
|
serverThread.setDaemon(true);
|
|
serverThread.start();
|
|
}
|
|
|
|
@Override
|
|
public void close() {
|
|
UCore.setPrivate(server, "shutdown", true);
|
|
connections.clear();
|
|
lastconnection = 0;
|
|
|
|
async(server::close);
|
|
}
|
|
|
|
@Override
|
|
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 = 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 = 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 ++){
|
|
connections.get(i).send(object, mode);
|
|
}
|
|
}
|
|
|
|
@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);
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void dispose(){
|
|
close();
|
|
Log.info("Disposed server.");
|
|
}
|
|
|
|
private void handleException(Throwable e){
|
|
Timers.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){
|
|
return con;
|
|
}
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
void async(Runnable run){
|
|
Thread thread = new Thread(run);
|
|
thread.setDaemon(true);
|
|
thread.start();
|
|
}
|
|
|
|
class KryoConnection extends NetConnection{
|
|
public final Connection connection;
|
|
|
|
public KryoConnection(int id, String address, Connection connection) {
|
|
super(id, address);
|
|
this.connection = connection;
|
|
}
|
|
|
|
@Override
|
|
public boolean isConnected(){
|
|
return connection.isConnected();
|
|
}
|
|
|
|
@Override
|
|
public void send(Object object, SendMode mode){
|
|
try {
|
|
if (mode == SendMode.tcp) {
|
|
connection.sendTCP(object);
|
|
} else {
|
|
connection.sendUDP(object);
|
|
}
|
|
}catch (Exception e){
|
|
Log.err(e);
|
|
Log.info("Disconnecting invalid client!");
|
|
connection.close();
|
|
|
|
KryoConnection k = getByKryoID(connection.getID());
|
|
if(k != null) connections.remove(k);
|
|
Log.info("Connection removed {0}", k);
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void close(){
|
|
if(connection.isConnected()) connection.close();
|
|
}
|
|
}
|
|
|
|
}
|