Compare commits

..

No commits in common. "7f3ec2276faeddcd83aa91e2ac202c9ad6dbb165" and "33aeb356a4e63e905e0b61508432ea59fc1c1c93" have entirely different histories.

27 changed files with 81 additions and 161 deletions

View file

@ -15,7 +15,7 @@
<dependencies> <dependencies>
<dependency> <dependency>
<groupId>de.hhhammer.dchat</groupId> <groupId>de.hhhammer.dchat</groupId>
<artifactId>discord-ws</artifactId> <artifactId>discord</artifactId>
<version>1.0-SNAPSHOT</version> <version>1.0-SNAPSHOT</version>
</dependency> </dependency>
<dependency> <dependency>

View file

@ -1,6 +1,5 @@
package de.hhhammer.dchat.bot; package de.hhhammer.dchat.bot;
import de.hhhammer.dchat.discord.ws.EventHandler;
import org.slf4j.Logger; import org.slf4j.Logger;
import org.slf4j.LoggerFactory; import org.slf4j.LoggerFactory;
@ -8,12 +7,6 @@ public final class App {
private static final Logger logger = LoggerFactory.getLogger(App.class); private static final Logger logger = LoggerFactory.getLogger(App.class);
public static void main(final String[] args) { public static void main(final String[] args) {
final String discordApiKey = System.getenv("DISCORD_API_KEY");
if (discordApiKey == null) {
logger.error("Missing environment variables: DISCORD_API_KEY");
System.exit(1);
}
final var eventHandler = new EventHandler.LogEventHandler();
logger.error("Currently not implemented!"); logger.error("Currently not implemented!");
} }
} }

View file

@ -1,4 +0,0 @@
module de.hhhammer.dchat.bot {
requires de.hhhammer.dchat.discord.ws;
requires org.slf4j;
}

View file

@ -1,23 +0,0 @@
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns="http://maven.apache.org/POM/4.0.0"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<parent>
<groupId>de.hhhammer.dchat</groupId>
<artifactId>dchat</artifactId>
<version>1.0-SNAPSHOT</version>
</parent>
<artifactId>discord-rest</artifactId>
<name>discord-rest</name>
<version>1.0-SNAPSHOT</version>
<packaging>jar</packaging>
<dependencies>
<dependency>
<groupId>org.json</groupId>
<artifactId>json</artifactId>
</dependency>
</dependencies>
</project>

View file

@ -1,7 +0,0 @@
package de.hhhammer.dchat.discord.rest;
public class Main {
public static void main(String[] args) {
System.out.println("Hello world!");
}
}

View file

@ -1,7 +0,0 @@
module de.hhhammer.dchat.discord.rest {
requires java.net.http;
requires org.json;
requires org.slf4j;
}

View file

@ -1,37 +0,0 @@
package de.hhhammer.dchat.discord.ws;
import de.hhhammer.dchat.discord.ws.connection.BackoffRetryer;
import de.hhhammer.dchat.discord.ws.connection.ConnectionManager;
import de.hhhammer.dchat.discord.ws.connection.Connector;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
public final class DiscordWebSocket {
private static final Logger logger = LoggerFactory.getLogger(DiscordWebSocket.class);
private static final String DISCORD_API_URL = "wss://gateway.discord.gg/?v=10&encoding=json";
private final String discordApiKey;
private final EventHandler eventHandler;
private final Retryer retryer;
public DiscordWebSocket(final String discordApiKey, final EventHandler eventHandler) {
this.discordApiKey = discordApiKey;
this.eventHandler = eventHandler;
this.retryer = new BackoffRetryer();
}
public static void main(final String[] args) throws InterruptedException {
final String discordApiKey = System.getenv("DISCORD_API_KEY");
if (discordApiKey == null) {
logger.error("Missing environment variables: DISCORD_API_KEY");
System.exit(1);
}
final var eventHandler = new EventHandler.LogEventHandler();
final var instance = new DiscordWebSocket(discordApiKey, eventHandler);
instance.start();
}
public void start() {
final var connector = new Connector(eventHandler);
final var connectionManager = new ConnectionManager(DISCORD_API_URL, discordApiKey, retryer);
}
}

View file

@ -1,7 +0,0 @@
package de.hhhammer.dchat.discord.ws;
public interface Retryer {
public int nextRetryInSeconds();
public boolean hasRetriesLeft();
}

View file

@ -1,8 +0,0 @@
module de.hhhammer.dchat.discord.ws {
exports de.hhhammer.dchat.discord.ws;
requires java.net.http;
requires org.json;
requires org.slf4j;
}

View file

@ -7,8 +7,8 @@
<version>1.0-SNAPSHOT</version> <version>1.0-SNAPSHOT</version>
</parent> </parent>
<artifactId>discord-ws</artifactId> <artifactId>discord</artifactId>
<name>discord-ws</name> <name>discord</name>
<version>1.0-SNAPSHOT</version> <version>1.0-SNAPSHOT</version>
<packaging>jar</packaging> <packaging>jar</packaging>
@ -17,5 +17,14 @@
<groupId>org.json</groupId> <groupId>org.json</groupId>
<artifactId>json</artifactId> <artifactId>json</artifactId>
</dependency> </dependency>
<dependency>
<groupId>org.junit.jupiter</groupId>
<artifactId>junit-jupiter</artifactId>
<scope>test</scope>
</dependency>
<dependency>
<groupId>ch.qos.logback</groupId>
<artifactId>logback-classic</artifactId>
</dependency>
</dependencies> </dependencies>
</project> </project>

View file

@ -1,4 +1,4 @@
package de.hhhammer.dchat.discord.ws.connection; package de.hhhammer.dchat;
public record ConnectionConfig(String gatewayUrl, ConnectionInitiator connectionInitiator) { public record ConnectionConfig(String gatewayUrl, ConnectionInitiator connectionInitiator) {
} }

View file

@ -1,4 +1,4 @@
package de.hhhammer.dchat.discord.ws.connection; package de.hhhammer.dchat;
import java.net.http.WebSocket; import java.net.http.WebSocket;

View file

@ -1,10 +1,9 @@
package de.hhhammer.dchat.discord.ws.connection; package de.hhhammer.dchat;
import de.hhhammer.dchat.discord.ws.Retryer; import de.hhhammer.dchat.model.CloseEvent;
import de.hhhammer.dchat.discord.ws.connection.event.CloseEvent;
import java.time.Duration; import java.time.Duration;
import java.util.function.Function;
public final class ConnectionManager { public final class ConnectionManager {
private final String initGatewayUrl; private final String initGatewayUrl;
@ -17,7 +16,7 @@ public final class ConnectionManager {
this.retryer = retryer; this.retryer = retryer;
} }
public void start(final Connector connector) throws InterruptedException { public void start(final Function<ConnectionConfig, CloseEvent> connector) throws InterruptedException {
while (retryer.hasRetriesLeft()) { while (retryer.hasRetriesLeft()) {
connect(connector); connect(connector);
final int reconnectDelayInSeconds = retryer.nextRetryInSeconds(); final int reconnectDelayInSeconds = retryer.nextRetryInSeconds();
@ -25,11 +24,11 @@ public final class ConnectionManager {
} }
} }
private void connect(final Connector connector) { private void connect(final Function<ConnectionConfig, CloseEvent> connector) {
ConnectionConfig mutConnectionConfig = new ConnectionConfig(initGatewayUrl, new IdendificationConnectionInitiator(token)); ConnectionConfig mutConnectionConfig = new ConnectionConfig(initGatewayUrl, new IdendificationConnectionInitiator(token));
boolean isResumable = true; boolean isResumable = true;
while (isResumable) { while (isResumable) {
final CloseEvent closeEvent = connector.connect(mutConnectionConfig); final CloseEvent closeEvent = connector.apply(mutConnectionConfig);
isResumable = switch (closeEvent) { isResumable = switch (closeEvent) {
case CloseEvent.ResumableCloseEvent resumableCloseEvent -> { case CloseEvent.ResumableCloseEvent resumableCloseEvent -> {
mutConnectionConfig = new ConnectionConfig( mutConnectionConfig = new ConnectionConfig(

View file

@ -1,21 +1,21 @@
package de.hhhammer.dchat.discord.ws.connection; package de.hhhammer.dchat;
import de.hhhammer.dchat.discord.ws.EventHandler; import de.hhhammer.dchat.model.CloseEvent;
import de.hhhammer.dchat.discord.ws.connection.event.CloseEvent;
import de.hhhammer.dchat.discord.ws.connection.listener.DiscordListener;
import java.net.URI; import java.net.URI;
import java.net.http.HttpClient; import java.net.http.HttpClient;
import java.util.concurrent.ArrayBlockingQueue; import java.util.concurrent.ArrayBlockingQueue;
import java.util.function.Function;
public final class Connector { public final class Connector implements Function<ConnectionConfig, CloseEvent> {
private final EventHandler eventHandler; private final EventHandler eventHandler;
public Connector(final EventHandler eventHandler) { public Connector(final EventHandler eventHandler) {
this.eventHandler = eventHandler; this.eventHandler = eventHandler;
} }
public CloseEvent connect(final ConnectionConfig connectionConfig) { @Override
public CloseEvent apply(final ConnectionConfig connectionConfig) {
final ArrayBlockingQueue<CloseEvent> closeEventQueue = new ArrayBlockingQueue<>(1); final ArrayBlockingQueue<CloseEvent> closeEventQueue = new ArrayBlockingQueue<>(1);
try (final HttpClient client = HttpClient.newHttpClient()) { try (final HttpClient client = HttpClient.newHttpClient()) {
client.newWebSocketBuilder() client.newWebSocketBuilder()

View file

@ -0,0 +1,22 @@
package de.hhhammer.dchat;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
public final class DiscordBotWebSocket {
private static final Logger logger = LoggerFactory.getLogger(DiscordBotWebSocket.class);
private static final String DISCORD_API_URL = "wss://gateway.discord.gg/?v=10&encoding=json";
public static void main(final String[] args) throws InterruptedException {
final String discordApiKey = System.getenv("DISCORD_API_KEY");
if (discordApiKey == null) {
logger.error("Missing environment variables: DISCORD_API_KEY");
System.exit(1);
}
final var eventHandler = new EventHandler.LogEventHandler();
final var connector = new Connector(eventHandler);
final var retryer = new Retryer();
final var connectionManager = new ConnectionManager(DISCORD_API_URL, discordApiKey, retryer);
connectionManager.start(connector);
}
}

View file

@ -1,11 +1,8 @@
package de.hhhammer.dchat.discord.ws.connection.listener; package de.hhhammer.dchat;
import de.hhhammer.dchat.discord.ws.EventHandler; import de.hhhammer.dchat.model.CloseEvent;
import de.hhhammer.dchat.discord.ws.connection.ConnectionInitiator; import de.hhhammer.dchat.model.Event;
import de.hhhammer.dchat.discord.ws.connection.event.CloseEvent; import de.hhhammer.dchat.model.EventType;
import de.hhhammer.dchat.discord.ws.connection.event.Event;
import de.hhhammer.dchat.discord.ws.connection.event.EventDeserializer;
import de.hhhammer.dchat.discord.ws.connection.event.EventType;
import org.json.JSONObject; import org.json.JSONObject;
import org.slf4j.Logger; import org.slf4j.Logger;
import org.slf4j.LoggerFactory; import org.slf4j.LoggerFactory;
@ -33,7 +30,7 @@ public final class DiscordListener implements WebSocket.Listener {
private final EventHandler eventHandler; private final EventHandler eventHandler;
private final BlockingQueue<CloseEvent> closeEventQueue; private final BlockingQueue<CloseEvent> closeEventQueue;
DiscordListener(final TextCollector textCollector, public DiscordListener(final TextCollector textCollector,
final EventDeserializer eventDeserializer, final EventDeserializer eventDeserializer,
final ConnectionInitiator connectionInitiator, final ConnectionInitiator connectionInitiator,
final EventHandler eventHandler, final EventHandler eventHandler,

View file

@ -1,5 +1,7 @@
package de.hhhammer.dchat.discord.ws.connection.event; package de.hhhammer.dchat;
import de.hhhammer.dchat.model.Event;
import de.hhhammer.dchat.model.EventType;
import org.json.JSONException; import org.json.JSONException;
import org.json.JSONObject; import org.json.JSONObject;

View file

@ -1,4 +1,4 @@
package de.hhhammer.dchat.discord.ws; package de.hhhammer.dchat;
import org.json.JSONObject; import org.json.JSONObject;
import org.slf4j.Logger; import org.slf4j.Logger;

View file

@ -1,4 +1,4 @@
package de.hhhammer.dchat.discord.ws.connection; package de.hhhammer.dchat;
import java.net.http.WebSocket; import java.net.http.WebSocket;

View file

@ -1,4 +1,4 @@
package de.hhhammer.dchat.discord.ws.connection; package de.hhhammer.dchat;
import java.net.http.WebSocket; import java.net.http.WebSocket;

View file

@ -1,22 +1,12 @@
package de.hhhammer.dchat.discord.ws.connection; package de.hhhammer.dchat;
import de.hhhammer.dchat.discord.ws.Retryer;
import java.util.concurrent.atomic.AtomicInteger; import java.util.concurrent.atomic.AtomicInteger;
public final class BackoffRetryer implements Retryer { public final class Retryer {
private static final int maxRetries = 5;
private static final int delayInSeconds = 10; private static final int delayInSeconds = 10;
private static final int backoffMultiplier = 2; private static final int backoffMultiplier = 2;
private final AtomicInteger tries = new AtomicInteger(); private final AtomicInteger tries = new AtomicInteger();
private final int maxRetries;
public BackoffRetryer(final int maxRetries) {
this.maxRetries = maxRetries;
}
public BackoffRetryer() {
this(5);
}
public int nextRetryInSeconds() { public int nextRetryInSeconds() {
final int currentTry = tries.getAndIncrement(); final int currentTry = tries.getAndIncrement();

View file

@ -1,11 +1,11 @@
package de.hhhammer.dchat.discord.ws.connection.listener; package de.hhhammer.dchat;
import org.slf4j.Logger; import org.slf4j.Logger;
import org.slf4j.LoggerFactory; import org.slf4j.LoggerFactory;
import java.util.Optional; import java.util.Optional;
final class TextCollector { public final class TextCollector {
private static final Logger logger = LoggerFactory.getLogger(TextCollector.class); private static final Logger logger = LoggerFactory.getLogger(TextCollector.class);
private final StringBuilder stringBuilder = new StringBuilder(); private final StringBuilder stringBuilder = new StringBuilder();

View file

@ -1,4 +1,4 @@
package de.hhhammer.dchat.discord.ws.connection.event; package de.hhhammer.dchat.model;
public sealed interface CloseEvent { public sealed interface CloseEvent {
record UnresumableCloseEvent() implements CloseEvent { record UnresumableCloseEvent() implements CloseEvent {

View file

@ -1,4 +1,4 @@
package de.hhhammer.dchat.discord.ws.connection.event; package de.hhhammer.dchat.model;
import org.json.JSONObject; import org.json.JSONObject;

View file

@ -1,4 +1,4 @@
package de.hhhammer.dchat.discord.ws.connection.event; package de.hhhammer.dchat.model;
public sealed interface EventType { public sealed interface EventType {
String value(); String value();

View file

@ -1,4 +1,4 @@
package de.hhhammer.dchat.discord.ws; package de.hhhammer.dchat;
import org.json.JSONObject; import org.json.JSONObject;
import org.junit.jupiter.api.Test; import org.junit.jupiter.api.Test;

15
pom.xml
View file

@ -42,11 +42,6 @@
<groupId>org.jetbrains</groupId> <groupId>org.jetbrains</groupId>
<artifactId>annotations</artifactId> <artifactId>annotations</artifactId>
</dependency> </dependency>
<dependency>
<groupId>org.junit.jupiter</groupId>
<artifactId>junit-jupiter</artifactId>
<scope>test</scope>
</dependency>
</dependencies> </dependencies>
@ -81,6 +76,13 @@
<artifactId>json</artifactId> <artifactId>json</artifactId>
<version>20240303</version> <version>20240303</version>
</dependency> </dependency>
<dependency>
<groupId>com.fasterxml.jackson</groupId>
<artifactId>jackson-bom</artifactId>
<version>2.18.1</version>
<type>pom</type>
<scope>import</scope>
</dependency>
</dependencies> </dependencies>
</dependencyManagement> </dependencyManagement>
<build> <build>
@ -96,7 +98,6 @@
</build> </build>
<modules> <modules>
<module>bot</module> <module>bot</module>
<module>discord-ws</module> <module>discord</module>
<module>discord-rest</module>
</modules> </modules>
</project> </project>