163 lines
5.2 KiB
Java
163 lines
5.2 KiB
Java
package fr.titionfire.ffsaf.net2;
|
|
|
|
import com.fasterxml.jackson.databind.JsonNode;
|
|
import fr.titionfire.ffsaf.domain.service.ClubService;
|
|
import fr.titionfire.ffsaf.domain.service.MembreService;
|
|
import fr.titionfire.ffsaf.net2.packet.IAction;
|
|
import fr.titionfire.ffsaf.net2.packet.RegisterAction;
|
|
import fr.titionfire.ffsaf.utils.Pair;
|
|
import io.quarkus.runtime.ShutdownEvent;
|
|
import io.quarkus.runtime.StartupEvent;
|
|
import jakarta.enterprise.event.Observes;
|
|
import jakarta.inject.Inject;
|
|
import jakarta.inject.Singleton;
|
|
import org.eclipse.microprofile.config.inject.ConfigProperty;
|
|
import org.jboss.logging.Logger;
|
|
import org.quartz.Scheduler;
|
|
|
|
import javax.net.ssl.*;
|
|
import java.io.File;
|
|
import java.io.FileInputStream;
|
|
import java.io.IOException;
|
|
import java.io.InputStream;
|
|
import java.net.Socket;
|
|
import java.security.*;
|
|
import java.util.ArrayList;
|
|
import java.util.HashMap;
|
|
import java.util.UUID;
|
|
import java.util.concurrent.ConcurrentHashMap;
|
|
import java.util.stream.Stream;
|
|
|
|
@Singleton
|
|
public class ServerCustom extends Thread {
|
|
private static final Logger LOGGER = Logger.getLogger("SocketServer");
|
|
|
|
protected ConcurrentHashMap<UUID, Pair<Client_Thread, Message<JsonNode>>> jobData = new ConcurrentHashMap<>();
|
|
protected HashMap<String, IAction> iMap = new HashMap<>();
|
|
public final ArrayList<Client_Thread> clients = new ArrayList<>();
|
|
|
|
private SSLServerSocket server;
|
|
private boolean run;
|
|
|
|
private static ServerCustom serverCustom;
|
|
|
|
@ConfigProperty(name = "internal-port")
|
|
int port;
|
|
|
|
@ConfigProperty(name = "certificate_password")
|
|
String certificate_password;
|
|
|
|
@ConfigProperty(name = "certificate_file")
|
|
String certificate_file;
|
|
|
|
@Inject
|
|
protected Scheduler quartz;
|
|
|
|
@Inject
|
|
public MembreService membreService;
|
|
|
|
@Inject
|
|
public ClubService clubService;
|
|
|
|
private PublicKey publicKey;
|
|
|
|
private X509TrustManager tm(KeyStore keystore) throws NoSuchAlgorithmException, KeyStoreException {
|
|
TrustManagerFactory trustMgrFactory = TrustManagerFactory.getInstance(TrustManagerFactory.getDefaultAlgorithm());
|
|
trustMgrFactory.init(keystore);
|
|
TrustManager[] trustManagers = trustMgrFactory.getTrustManagers();
|
|
for (TrustManager trustManager : trustManagers) {
|
|
if (trustManager instanceof X509TrustManager) {
|
|
return (X509TrustManager) trustManager;
|
|
}
|
|
}
|
|
return null;
|
|
}
|
|
|
|
private X509KeyManager km(KeyStore keystore) throws NoSuchAlgorithmException, KeyStoreException, UnrecoverableKeyException {
|
|
KeyManagerFactory keyMgrFactory = KeyManagerFactory.getInstance(KeyManagerFactory.getDefaultAlgorithm());
|
|
keyMgrFactory.init(keystore, certificate_password.toCharArray());
|
|
KeyManager[] keyManagers = keyMgrFactory.getKeyManagers();
|
|
for (KeyManager keyManager : keyManagers) {
|
|
if (keyManager instanceof X509KeyManager) {
|
|
return (X509KeyManager) keyManager;
|
|
}
|
|
}
|
|
return null;
|
|
}
|
|
|
|
private SSLServerSocket getServerSocketWithCert(int port, InputStream pathToCert) throws Exception {
|
|
TrustManager[] tmm = new TrustManager[1];
|
|
KeyManager[] kmm = new KeyManager[1];
|
|
|
|
KeyStore ks = KeyStore.getInstance("JKS");
|
|
ks.load(pathToCert, certificate_password.toCharArray());
|
|
tmm[0] = tm(ks);
|
|
kmm[0] = km(ks);
|
|
|
|
SSLContext ctx = SSLContext.getInstance("TLSv1.2");
|
|
ctx.init(kmm, tmm, null);
|
|
|
|
publicKey = ks.getCertificate("ffsaf").getPublicKey();
|
|
|
|
SSLServerSocketFactory socketFactory = ctx.getServerSocketFactory();
|
|
return (SSLServerSocket) socketFactory.createServerSocket(port);
|
|
}
|
|
|
|
@Override
|
|
public void run() {
|
|
try {
|
|
this.run = true;
|
|
this.server = getServerSocketWithCert(this.port, new FileInputStream(new File(this.certificate_file)));
|
|
RegisterAction.register(iMap);
|
|
LOGGER.info("Starting socket server!!");
|
|
while (this.run) {
|
|
Socket socket = this.server.accept();
|
|
Client_Thread cThread = new Client_Thread(this, socket, publicKey);
|
|
synchronized (clients) {
|
|
this.clients.add(cThread);
|
|
}
|
|
cThread.start();
|
|
}
|
|
LOGGER.info("Shutting down Socket server!!");
|
|
} catch (Exception e) {
|
|
LOGGER.info("Server start failed! (" + e.getMessage() + ")");
|
|
this.interrupt();
|
|
}
|
|
}
|
|
|
|
public void onCloseClient(Client_Thread cThread) {
|
|
synchronized (clients) {
|
|
clients.remove(cThread);
|
|
}
|
|
}
|
|
|
|
public void stop2() {
|
|
this.run = false;
|
|
|
|
try {
|
|
this.server.close();
|
|
clients.forEach(Client_Thread::stopSoket);
|
|
} catch (IOException ignored) {
|
|
}
|
|
}
|
|
|
|
void onStart(@Observes StartupEvent ev) {
|
|
LOGGER.info("The server is starting...");
|
|
serverCustom = this;
|
|
this.start();
|
|
}
|
|
|
|
void onStop(@Observes ShutdownEvent ev) {
|
|
LOGGER.info("The server is stopping...");
|
|
this.stop2();
|
|
}
|
|
|
|
public static ServerCustom getInstance(){
|
|
return serverCustom;
|
|
}
|
|
|
|
public Stream<Client_Thread> get_clients() {
|
|
return clients.stream();
|
|
}
|
|
}
|