SubServers-2/SubServers.Bungee/src/net/ME1312/SubServers/Bungee/Host/External/ExternalHost.java

328 lines
13 KiB
Java
Raw Normal View History

package net.ME1312.SubServers.Bungee.Host.External;
2020-11-16 21:34:59 +01:00
import net.ME1312.Galaxi.Library.Map.ObjectMap;
import net.ME1312.Galaxi.Library.Try;
2020-11-16 21:34:59 +01:00
import net.ME1312.Galaxi.Library.Util;
2019-05-10 04:43:34 +02:00
import net.ME1312.SubData.Server.ClientHandler;
2019-04-18 16:02:09 +02:00
import net.ME1312.SubData.Server.DataClient;
import net.ME1312.SubData.Server.Protocol.PacketOut;
2019-04-18 16:02:09 +02:00
import net.ME1312.SubData.Server.SubDataClient;
import net.ME1312.SubServers.Bungee.Event.SubAddServerEvent;
import net.ME1312.SubServers.Bungee.Event.SubRemoveServerEvent;
import net.ME1312.SubServers.Bungee.Host.Host;
import net.ME1312.SubServers.Bungee.Host.SubCreator;
import net.ME1312.SubServers.Bungee.Host.SubServer;
2019-05-17 00:36:41 +02:00
import net.ME1312.SubServers.Bungee.Library.Compatibility.Logger;
import net.ME1312.SubServers.Bungee.Library.Exception.InvalidServerException;
import net.ME1312.SubServers.Bungee.Network.Packet.PacketExAddServer;
2017-04-01 18:50:09 +02:00
import net.ME1312.SubServers.Bungee.Network.Packet.PacketExDeleteServer;
import net.ME1312.SubServers.Bungee.Network.Packet.PacketExRemoveServer;
2019-04-18 16:02:09 +02:00
import net.ME1312.SubServers.Bungee.Network.Packet.PacketOutExReset;
import net.ME1312.SubServers.Bungee.SubProxy;
2020-11-16 21:34:59 +01:00
import com.google.common.collect.Range;
import java.net.InetAddress;
import java.util.*;
2017-04-16 19:02:14 +02:00
/**
* External Host Class
*/
2019-05-10 04:43:34 +02:00
public class ExternalHost extends Host implements ClientHandler {
private HashMap<Integer, SubDataClient> subdata = new HashMap<Integer, SubDataClient>();
private HashMap<String, SubServer> servers = new HashMap<String, SubServer>();
private String name;
boolean available;
private boolean enabled;
private InetAddress address;
private SubCreator creator;
private String directory;
private LinkedList<PacketOut> queue;
2017-04-01 22:31:57 +02:00
private boolean clean;
SubProxy plugin;
/**
* Creates an External Host
*
2019-01-05 22:04:45 +01:00
* @param plugin SubServers Internals
* @param name The Name of your Host
* @param ports The range of ports to auto-select from
* @param log Whether apps like SubCreator should log to console (does not apply to servers)
* @param enabled If your host is Enabled
* @param address The address of your Host
* @param directory The runtime directory of your Host
* @param gitBash The Git Bash directory
*/
public ExternalHost(SubProxy plugin, String name, boolean enabled, Range<Integer> ports, boolean log, InetAddress address, String directory, String gitBash) {
2019-01-05 22:04:45 +01:00
super(plugin, name, enabled, ports, log, address, directory, gitBash);
this.plugin = plugin;
this.name = name;
2019-05-10 04:43:34 +02:00
this.available = false;
this.enabled = enabled;
this.address = address;
2019-01-05 22:04:45 +01:00
this.creator = new ExternalSubCreator(this, ports, log, gitBash);
this.directory = directory;
this.queue = new LinkedList<PacketOut>();
2017-04-01 22:31:57 +02:00
this.clean = false;
2019-05-10 04:43:34 +02:00
2019-05-14 04:02:38 +02:00
subdata.put(0, null);
}
@Override
2019-05-10 04:43:34 +02:00
public DataClient[] getSubData() {
2021-04-08 15:15:35 +02:00
Integer[] keys = subdata.keySet().toArray(new Integer[0]);
DataClient[] channels = new DataClient[keys.length];
Arrays.sort(keys);
for (int i = 0; i < keys.length; ++i) channels[i] = subdata.get(keys[i]);
return channels;
2019-05-10 04:43:34 +02:00
}
public void setSubData(DataClient client, int channel) {
if (channel < 0) throw new IllegalArgumentException("Subchannel ID cannot be less than zero");
2022-02-14 18:53:55 +01:00
if (client == null && channel == 0) available = false;
2019-05-10 06:04:10 +02:00
if (client != null || channel == 0) {
if (!subdata.containsKey(channel) || (channel == 0 && (client == null || subdata.get(channel) == null))) {
2019-05-10 04:43:34 +02:00
subdata.put(channel, (SubDataClient) client);
if (client != null && (client.getHandler() == null || !equals(client.getHandler()))) ((SubDataClient) client).setHandler(this);
}
2019-05-10 06:04:10 +02:00
} else {
subdata.remove(channel);
2019-05-10 04:43:34 +02:00
}
}
@Override
2019-05-10 04:43:34 +02:00
public void removeSubData(DataClient client) {
for (Integer channel : Util.getBackwards(subdata, (SubDataClient) client)) setSubData(null, channel);
}
void queue(PacketOut... packet) {
for (PacketOut p : packet) if (getSubData()[0] == null || !available) {
queue.add(p);
} else {
2019-05-10 04:43:34 +02:00
((SubDataClient) getSubData()[0]).sendPacket(p);
}
}
private void requeue() {
2019-05-10 04:43:34 +02:00
SubDataClient client = (SubDataClient) getSubData()[0];
2017-04-01 22:31:57 +02:00
if (!clean) {
2019-05-10 04:43:34 +02:00
client.sendPacket(new PacketOutExReset("Prevent Desync"));
2017-04-01 22:31:57 +02:00
clean = true;
}
2022-02-14 18:53:55 +01:00
HashSet<String> served = new HashSet<String>();
LinkedList<PacketOut> queue = this.queue; this.queue = new LinkedList<PacketOut>();
PacketOut[] payload = new PacketOut[queue.size()];
for (int i = 0; i < payload.length; ++i) {
PacketOut packet = queue.get(i);
if (packet instanceof PacketExAddServer) served.add(((PacketExAddServer) packet).peek());
payload[i] = packet;
}
2022-02-14 18:53:55 +01:00
for (SubServer server : servers.values()) {
if (!served.contains(server.getName())) {
client.sendPacket(new PacketExAddServer((ExternalSubServer) server, (server.isRunning())?((ExternalSubLogger) server.getLogger()).getExternalAddress():null, data -> {
if (data.contains(0x0002)) ((ExternalSubServer) server).started(data.getUUID(0x0002));
}));
}
}
2022-02-14 18:53:55 +01:00
client.sendPacket(payload);
2019-05-10 04:43:34 +02:00
available = true;
2022-02-14 18:53:55 +01:00
while (this.queue.size() != 0) {
client.sendPacket(this.queue.remove(0));
}
}
@Override
public boolean isAvailable() {
2019-05-10 04:43:34 +02:00
return available;
}
@Override
public boolean isEnabled() {
return enabled;
}
@Override
public void setEnabled(boolean value) {
this.enabled = value;
}
@Override
public InetAddress getAddress() {
return address;
}
@Override
2017-04-24 18:28:16 +02:00
public String getPath() {
return directory;
}
@Override
public String getName() {
return name;
}
@Override
public SubCreator getCreator() {
return creator;
}
@Override
public Map<String, ? extends SubServer> getSubServers() {
return new TreeMap<String, SubServer>(servers);
}
@Override
public SubServer getSubServer(String name) {
2021-06-15 06:09:51 +02:00
if (Util.isNull(name)) return null;
return servers.get(name.toLowerCase());
}
@Override
public SubServer constructSubServer(String name, boolean enabled, int port, String motd, boolean log, String directory, String executable, String stopcmd, boolean hidden, boolean restricted) throws InvalidServerException {
return ExternalSubServer.construct(this, name, enabled, port, motd, log, directory, executable, stopcmd, hidden, restricted);
}
@Override
public boolean addSubServer(UUID player, SubServer server) throws InvalidServerException {
if (server.getHost() != this) throw new IllegalArgumentException("That Server does not belong to this Host!");
2021-06-06 08:02:27 +02:00
if (plugin.api.getServers().containsKey(server.getName().toLowerCase())) throw new InvalidServerException("A Server already exists with this name!");
SubAddServerEvent event = new SubAddServerEvent(player, this, server);
plugin.getPluginManager().callEvent(event);
if (!event.isCancelled()) {
queue(new PacketExAddServer(((ExternalSubServer) server), (server.isRunning())?((ExternalSubLogger) server.getLogger()).getExternalAddress():null, data -> {
if (data.contains(0x0002)) ((ExternalSubServer) server).started(data.getUUID(0x0002));
2022-02-15 17:43:25 +01:00
((ExternalSubServer) server).registered(true);
}));
servers.put(server.getName().toLowerCase(), server);
return true;
} else {
return false;
}
}
@Override
2020-08-14 00:26:22 +02:00
protected boolean removeSubServer(UUID player, String name, boolean forced) throws InterruptedException {
Util.nullpo(name);
ExternalSubServer server = (ExternalSubServer) servers.get(name.toLowerCase());
SubRemoveServerEvent event = new SubRemoveServerEvent(player, this, server);
plugin.getPluginManager().callEvent(event);
2020-08-14 00:26:22 +02:00
if (forced || !event.isCancelled()) {
server.registered(false);
if (server.isRunning()) {
server.stop();
server.waitFor();
}
servers.remove(name.toLowerCase());
queue(new PacketExRemoveServer(name.toLowerCase(), data -> {
if (data.getInt(0x0001) != 0 && data.getInt(0x0001) != 1) {
server.registered(true);
servers.put(name.toLowerCase(), server);
2017-08-26 07:19:59 +02:00
}
}));
return true;
} else return false;
}
@Override
2020-08-14 00:26:22 +02:00
protected boolean recycleSubServer(UUID player, String name, boolean forced) throws InterruptedException {
Util.nullpo(name);
ExternalSubServer s = (ExternalSubServer) servers.get(name.toLowerCase());
String server = s.getName();
2019-04-18 16:02:09 +02:00
SubRemoveServerEvent event = new SubRemoveServerEvent(player, this, s);
2019-04-18 16:02:09 +02:00
plugin.getPluginManager().callEvent(event);
2020-08-14 00:26:22 +02:00
if (forced || !event.isCancelled()) {
s.registered(false);
if (s.isRunning()) {
s.stop();
s.waitFor();
2019-04-18 16:02:09 +02:00
}
2019-05-17 00:36:41 +02:00
Logger.get("SubServers").info("Saving...");
2019-04-29 17:05:06 +02:00
ObjectMap<String> info = (plugin.servers.get().getMap("Servers").getKeys().contains(server))?plugin.servers.get().getMap("Servers").getMap(server).clone():new ObjectMap<String>();
2019-04-18 16:02:09 +02:00
info.set("Name", server);
info.set("Timestamp", Calendar.getInstance().getTime().getTime());
try {
2019-04-29 17:05:06 +02:00
if (plugin.servers.get().getMap("Servers").getKeys().contains(server)) {
plugin.servers.get().getMap("Servers").remove(server);
plugin.servers.save();
2019-04-18 16:02:09 +02:00
}
} catch (Exception e) {
e.printStackTrace();
}
2019-05-17 00:36:41 +02:00
Logger.get("SubServers").info("Moving Files...");
2019-04-18 16:02:09 +02:00
queue(new PacketExDeleteServer(server, info, true, data -> {
2019-05-13 05:28:27 +02:00
if (data.getInt(0x0001) == 0 || data.getInt(0x0001) == 1) {
2019-04-18 16:02:09 +02:00
servers.remove(server.toLowerCase());
2019-05-17 00:36:41 +02:00
Logger.get("SubServers").info("Deleted SubServer: " + server);
2019-04-18 16:02:09 +02:00
} else {
s.registered(true);
2019-05-17 00:36:41 +02:00
Logger.get("SubServers").info("Couldn't remove " + server + " from memory. See " + getName() + " console for more details");
2019-04-18 16:02:09 +02:00
}
}));
return true;
} else return false;
}
@Override
2020-08-14 00:26:22 +02:00
protected boolean deleteSubServer(UUID player, String name, boolean forced) throws InterruptedException {
Util.nullpo(name);
ExternalSubServer s = (ExternalSubServer) servers.get(name.toLowerCase());
String server = s.getName();
2017-04-01 18:50:09 +02:00
SubRemoveServerEvent event = new SubRemoveServerEvent(player, this, getSubServer(server));
2017-04-01 18:50:09 +02:00
plugin.getPluginManager().callEvent(event);
2020-08-14 00:26:22 +02:00
if (forced || !event.isCancelled()) {
s.registered(false);
if (s.isRunning()) {
s.stop();
s.waitFor();
2017-04-01 18:50:09 +02:00
}
2019-05-17 00:36:41 +02:00
Logger.get("SubServers").info("Saving...");
2019-04-29 17:05:06 +02:00
ObjectMap<String> info = (plugin.servers.get().getMap("Servers").getKeys().contains(server))?plugin.servers.get().getMap("Servers").getMap(server).clone():new ObjectMap<String>();
info.set("Name", server);
info.set("Timestamp", Calendar.getInstance().getTime().getTime());
2017-04-01 18:50:09 +02:00
try {
2019-04-29 17:05:06 +02:00
if (plugin.servers.get().getMap("Servers").getKeys().contains(server)) {
plugin.servers.get().getMap("Servers").remove(server);
plugin.servers.save();
2017-04-01 18:50:09 +02:00
}
} catch (Exception e) {
e.printStackTrace();
}
2019-05-17 00:36:41 +02:00
Logger.get("SubServers").info("Removing Files...");
2019-04-18 16:02:09 +02:00
queue(new PacketExDeleteServer(server, info, false, data -> {
2019-05-13 05:28:27 +02:00
if (data.getInt(0x0001) == 0 || data.getInt(0x0001) == 1) {
servers.remove(server.toLowerCase());
2019-05-17 00:36:41 +02:00
Logger.get("SubServers").info("Deleted SubServer: " + server);
2017-04-01 18:50:09 +02:00
} else {
s.registered(true);
2019-05-17 00:36:41 +02:00
Logger.get("SubServers").info("Couldn't remove " + server + " from memory. See " + getName() + " console for more details");
2017-04-01 18:50:09 +02:00
}
}));
return true;
} else return false;
}
2017-04-24 18:28:16 +02:00
@Override
public boolean destroy() {
if (Try.all.get(() -> Util.reflect(SubProxy.class.getDeclaredField("running"), plugin), true)) {
return super.destroy();
}
return true;
}
2019-05-14 04:02:38 +02:00
@Override
public ObjectMap<String> forSubData() {
ObjectMap<String> hinfo = super.forSubData();
ObjectMap<Integer> subdata = new ObjectMap<Integer>();
for (int channel : this.subdata.keySet()) subdata.set(channel, (this.subdata.get(channel) == null)?null:this.subdata.get(channel).getID());
hinfo.set("subdata", subdata);
return hinfo;
}
}