dev.qixils.crowdcontrol.socket.RequestHandler Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of crowd-control-sender Show documentation
Show all versions of crowd-control-sender Show documentation
Library for sending effect requests to a receiving game server or clients
package dev.qixils.crowdcontrol.socket;
import com.google.gson.JsonParseException;
import dev.qixils.crowdcontrol.SimulatedService;
import dev.qixils.crowdcontrol.TriState;
import dev.qixils.crowdcontrol.exceptions.CrowdControlException;
import dev.qixils.crowdcontrol.exceptions.EffectUnavailableException;
import dev.qixils.crowdcontrol.exceptions.ExceptionUtil;
import org.jetbrains.annotations.Blocking;
import org.jetbrains.annotations.NotNull;
import org.jetbrains.annotations.Nullable;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import reactor.core.publisher.Flux;
import reactor.core.publisher.FluxSink;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.Socket;
import java.nio.charset.StandardCharsets;
import java.time.Duration;
import java.util.Map;
import java.util.Objects;
import java.util.concurrent.*;
final class RequestHandler implements SimulatedService {
private static final @NotNull Logger logger = LoggerFactory.getLogger("CrowdControl/RequestHandler");
private static final @NotNull Executor executor = Executors.newCachedThreadPool();
private static final @NotNull ScheduledExecutorService scheduledExecutor = Executors.newSingleThreadScheduledExecutor();
private final @NotNull Map effectDataMap = new ConcurrentHashMap<>(1);
private final @NotNull Map effectAvailabilityMap = new ConcurrentHashMap<>(1);
private final Socket socket;
private final SimulatedService> parent;
private final InputStream inputStream;
private final OutputStream outputStream;
private final @Nullable String encryptedPassword;
private final Thread loopThread;
private boolean running = true;
private int nextRequestId = 0;
private boolean loggedIn;
RequestHandler(@NotNull Socket socket, @NotNull SimulatedService> parent, @Nullable String encryptedPassword) throws IOException {
this.socket = ExceptionUtil.validateNotNull(socket, "socket");
this.parent = ExceptionUtil.validateNotNull(parent, "parent");
this.inputStream = socket.getInputStream();
this.outputStream = socket.getOutputStream();
this.encryptedPassword = encryptedPassword;
loggedIn = encryptedPassword == null;
loopThread = new Thread(this::loop);
}
public void start() throws IllegalThreadStateException {
logger.info("Starting request handler");
loopThread.start();
}
@Override
public void shutdown() {
if (!running) return;
running = false;
logger.info("Shutting down request handler");
try {
socket.close();
} catch (IOException e) {
logger.warn("Failed to close socket", e);
}
parent.shutdown();
// wait for request streams to finish
scheduledExecutor.schedule(() -> {
effectDataMap.forEach(($, data) -> data.sink.error(new CrowdControlException("RequestHandler shutting down")));
effectDataMap.clear();
}, 2, TimeUnit.SECONDS);
}
@Override
public boolean isRunning() {
return running && !socket.isClosed() && loopThread.isAlive();
}
@Override
public boolean isAcceptingRequests() {
return isRunning() && loggedIn;
}
@Override
public boolean isShutdown() {
return !running;
}
@Override
public @NotNull TriState isEffectAvailable(@Nullable String effect) {
if (effect == null)
return TriState.FALSE;
return TriState.fromBoolean(effectAvailabilityMap.get(effect));
}
@Blocking
private void loop() {
try {
while (running) {
Response response;
try {
response = JsonObject.fromInputStream(inputStream, Response::fromJSON);
} catch (JsonParseException e) {
logger.error("Failed to parse JSON from socket", e);
return;
}
if (response == null) {
// assuming socket is closed because we received an empty packet
shutdown();
return;
}
switch (response.getPacketType()) {
case LOGIN:
if (encryptedPassword == null)
throw new IllegalStateException("Service sent LOGIN packet, but no password was provided");
logger.info("Login prompted; sending password");
sendRequest(new Request.Builder()
.type(Request.Type.LOGIN)
.password(encryptedPassword)
).subscribe();
break;
case DISCONNECT:
logger.warn("Disconnected from service: " + response.getMessage());
shutdown();
break;
case LOGIN_SUCCESS:
logger.info("Login successful");
loggedIn = true;
break;
case EFFECT_RESULT:
EffectData data = effectDataMap.get(response.getId());
if (data == null) {
logger.debug("Received response for unknown request ID: " + response.getId());
break;
}
logger.debug("Received response for request " + response.getId());
data.responseReceived = true;
data.sink.next(response); // send response to subscriber
// set availability of effect
// TODO: this is silently erroring in unit tests
String effectName = data.request.getEffect();
if (!effectAvailabilityMap.containsKey(effectName))
effectAvailabilityMap.put(effectName, response.getResultType() != Response.ResultType.UNAVAILABLE);
boolean toSchedule = false; // whether to schedule a fake FINISHED response
// handle the various possible response types/states
// TODO: handle menu updates
if (response.isTerminating()) {
data.sink.complete();
} else if (response.getResultType() == Response.ResultType.RETRY) {
int retryDelay = data.getRetryDelay();
if (retryDelay == -1)
data.sink.complete();
else
scheduledExecutor.schedule(
() -> writeRequest(data.request, data.sink),
retryDelay, TimeUnit.SECONDS
);
} else if (response.getResultType() == Response.ResultType.PAUSED) {
data.pause();
} else if (response.getResultType() == Response.ResultType.RESUMED) {
data.resume();
toSchedule = true;
} else if (response.getResultType() == Response.ResultType.SUCCESS) {
// this represents the start of a timed effect
// because this was not caught by the first if block
data.updateTimeRemaining(response.getTimeRemaining());
toSchedule = true;
}
if (toSchedule)
data.scheduledFuture = scheduledExecutor.schedule(
() -> {
// send fake FINISHED packet to flux stream
data.sink.next(new Response.Builder()
.id(response.getId())
.type(Response.ResultType.FINISHED)
.build()
);
// complete flux stream
data.sink.complete();
},
ExceptionUtil.validateNotNullElse(response.getTimeRemaining(), Duration.ZERO).toMillis(),
TimeUnit.MILLISECONDS
);
}
}
} catch (IOException e) {
if (running)
logger.error("Failed to read from socket", e);
}
}
@Override
public @NotNull Flux<@NotNull Response> sendRequest(Request.@NotNull Builder builder, @Nullable Duration timeout) throws IllegalStateException {
ExceptionUtil.validateNotNull(builder, "builder");
Request.Type type = builder.type();
if (type == null)
throw new IllegalArgumentException("Request type is null");
// create request (done first to ensure it is valid (i.e. doesn't throw))
if (type.usesIncrementalIds() == TriState.TRUE)
builder.id(++nextRequestId);
Request request = builder.build();
// ensure effect is available
if (type.isEffectType() && isEffectAvailable(request.getEffect()) == TriState.FALSE)
throw new EffectUnavailableException("Effect " + request.getEffect() + " is known to be unavailable to this service");
return Flux.create(sink -> {
// ensure service is accepting requests
if (type.isEffectType() && !isAcceptingRequests()) {
sink.error(new IllegalStateException("RequestHandler is not accepting requests"));
return;
}
// add sink to map of publishers
EffectData data = new EffectData(request.getId(), request, sink);
effectDataMap.put(request.getId(), data);
// manage responseReceivedMap for timeout functionality
if (timeout != null) {
scheduledExecutor.schedule(() -> {
if (!isAcceptingRequests()) return;
if (data.responseReceived) return;
final String error = "Timed out waiting for response for request " + request.getId();
logger.debug(error);
sink.error(new TimeoutException(error));
}, timeout.toMillis(), TimeUnit.MILLISECONDS);
}
// send request
executor.execute(() -> writeRequest(request, sink));
}).doOnComplete(() -> effectDataMap.remove(request.getId()));
}
private void writeRequest(@NotNull Request request, @NotNull FluxSink sink) {
assert isAcceptingRequests() || (isRunning() && !request.getType().isEffectType());
try {
outputStream.write(request.toJSON().getBytes(StandardCharsets.UTF_8));
outputStream.write(0x00);
outputStream.flush();
} catch (Exception e) {
logger.warn("Failed to send request", e);
sink.error(e);
}
}
private static final class EffectData {
private final int id;
private final @NotNull Request request;
private final @NotNull FluxSink<@NotNull Response> sink;
private boolean responseReceived = false;
private int retryCount = 0;
private long timeRemaining = 0;
private long timeUpdatedAt = 0;
private boolean paused = false;
private @Nullable ScheduledFuture> scheduledFuture;
private EffectData(int id, @NotNull Request request, @NotNull FluxSink<@NotNull Response> sink) {
this.id = id;
this.request = request;
this.sink = sink;
}
private int getRetryDelay() {
if (retryCount > 6) return -1;
return (int) Math.pow(2, 2 + (retryCount++));
}
private void updateTimeRemaining(@Nullable Duration timeRemaining) {
this.timeRemaining = ExceptionUtil.validateNotNullElse(timeRemaining, Duration.ZERO).toMillis();
this.timeUpdatedAt = System.currentTimeMillis();
}
// get the elapsed time since the last update
@NotNull
private Duration getCurrentTimeRemaining() {
return Duration.ofMillis(Math.max(0, timeRemaining - (System.currentTimeMillis() - timeUpdatedAt)));
}
private void pause() {
if (paused) return;
paused = true;
if (scheduledFuture != null)
scheduledFuture.cancel(false);
scheduledFuture = null;
// update the time remaining using the elapsed time since the last update
updateTimeRemaining(getCurrentTimeRemaining());
}
private void resume() {
if (!paused) return;
paused = false;
timeUpdatedAt = System.currentTimeMillis();
}
@Override
public boolean equals(@Nullable Object o) {
if (this == o) return true;
if (o == null || getClass() != o.getClass()) return false;
EffectData that = (EffectData) o;
return id == that.id && responseReceived == that.responseReceived && retryCount == that.retryCount && sink.equals(that.sink);
}
@Override
public int hashCode() {
return Objects.hash(id, sink, responseReceived, retryCount);
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy