org.nustaq.kontraktor.services.rlserver.RLJsonServer Maven / Gradle / Ivy
package org.nustaq.kontraktor.services.rlserver;
import org.nustaq.kontraktor.Actor;
import org.nustaq.kontraktor.IPromise;
import org.nustaq.kontraktor.Scheduler;
import org.nustaq.kontraktor.annotations.CallerSideMethod;
import org.nustaq.kontraktor.annotations.Local;
import org.nustaq.kontraktor.apputil.*;
import org.nustaq.kontraktor.impl.BackOffStrategy;
import org.nustaq.kontraktor.impl.SimpleScheduler;
import org.nustaq.kontraktor.remoting.encoding.Coding;
import org.nustaq.kontraktor.remoting.encoding.SerializerType;
import org.nustaq.kontraktor.remoting.http.undertow.Http4K;
import org.nustaq.kontraktor.remoting.tcp.TCPPublisher;
import org.nustaq.kontraktor.services.rlclient.DataClient;
import org.nustaq.kontraktor.util.Log;
import org.nustaq.reallive.messages.*;
import org.nustaq.reallive.records.MapRecord;
import java.io.File;
import java.util.Random;
import java.util.concurrent.TimeUnit;
import java.util.stream.IntStream;
public class RLJsonServer extends Actor {
public static final Class[] CLAZZES = new Class[]{
LoginData.class,
MapRecord.class,
AddMessage.class,
UpdateMessage.class,
RemoveMessage.class,
QueryDoneMessage.class,
SessionEvent.class,
RLJsonAuthResult.class,
Diff.class,
};
public static SimpleRLConfig Cfg() {
synchronized (RLJsonServer.class) {
if ( cfg == null )
cfg = SimpleRLConfig.read();
return cfg;
}
}
private static SimpleRLConfig cfg;
// threads to dispatch session onto
protected Scheduler clientThreads[];
protected Random rand = new Random();
protected RLJsonServerService service;
protected DataClient dclient;
@Local
public IPromise init(String args[]) {
clientThreads = new Scheduler[Cfg().getNumSessionThreads()];
IntStream.range(0,Cfg().getNumSessionThreads())
.forEach( i -> clientThreads[i] = new SimpleScheduler(10000, true /*Important!*/ ));
service = RLJsonServerService.start(args);
service.setWebServer(self());
dclient = service.getDClient();
return resolve();
}
public IPromise authenticate(String user, String pwd ) {
Log.Info(this,"authenticate session");
return createSession(new Object[]{ user,pwd });
}
protected IPromise createSession(Object customSessionData) {
// FIXME, check credentials
RLJsonSession session = AsActor(
getSessionActorClazz(customSessionData),
// randomly distribute session actors among clientThreads
clientThreads[rand.nextInt(clientThreads.length)]
);
session.init(self(),dclient,customSessionData);
return resolve(new RLJsonAuthResult().session(session));
}
protected Class extends RLJsonSession> getSessionActorClazz(Object authData) {
return RLJsonSession.class;
}
@CallerSideMethod
protected void createServer(RLJsonServer app, Class[] CLAZZES) {
Http4K.Build(Cfg().getBindIp(), Cfg().getBindPort())
.httpAPI("/api", app) // could also be websocket based (see IntrinsicReactJSX github project)
.coding(new Coding(SerializerType.JsonNoRef, CLAZZES))
.setSessionTimeout(TimeUnit.MINUTES.toMillis(cfg.getSessionTimeoutMinutes() ))
.buildHttpApi()
.websocket("/ws", app)
.coding(new Coding(SerializerType.JsonNoRef, CLAZZES))
.buildWebsocket()
.build();
new TCPPublisher(app,Cfg().tcpPort)
// .coding( new Coding(SerializerType.JsonNoRef,RLJsonServer.CLAZZES) )
.publish( dis -> System.out.println("disconnected"));
}
@CallerSideMethod @Local
public DataClient getDClient() {
return getActor().dclient;
}
public static void main(String[] args) throws InterruptedException {
BackOffStrategy.SLEEP_NANOS = 5 * 1000 * 1000; // 5 millis
Class appClazz = RLJsonServer.class;
startUp(args, appClazz);
}
public static void startUp(String[] args, Class appClazz) throws InterruptedException {
if ( ! new File(SimpleRLConfig.pathname).exists() ) {
System.out.println("Please run with project working dir");
System.exit(-1);
}
if ( Cfg().runDataClusterInsideWebserver ) {
SingleProcessRLCluster.main(new String[0]);
}
RLJsonServer app = (RLJsonServer) AsActor(appClazz);
app.init(args).await(5*60_000);
Log.Info(appClazz,"listening on http://"+Cfg().getBindIp()+":"+Cfg().getBindPort());
app.createServer(app, CLAZZES);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy