Compare commits

..

2 commits

Author SHA1 Message Date
a51aea8013 discord-rest: Add minimal error handling 2024-11-19 20:38:56 +01:00
d15388581b openai-rest: Add OpenAI client
Missing the response deserialization.
2024-11-19 20:36:56 +01:00
9 changed files with 102 additions and 77 deletions

View file

@ -24,31 +24,31 @@ public class DiscordRest {
this.webConfig = webConfig;
}
public void postMessage(final int channelId, final Message message) throws IOException, InterruptedException {
final var uri = URI.create(DISCORD_URL + "/channels/%s/messages".formatted(channelId));
final HttpRequest request = HttpRequest.newBuilder()
public void postMessage(int channelId, Message message) throws IOException, InterruptedException {
var uri = URI.create(DISCORD_URL + "/channels/%s/messages".formatted(channelId));
HttpRequest request = HttpRequest.newBuilder()
.uri(uri)
.header("Authorization", "Bot " + webConfig.token())
.header("User-Agent", USER_AGENT)
.POST(HttpRequest.BodyPublishers.ofString(message.toJson()))
.build();
final HttpResponse<Void> response = httpClient.send(request, HttpResponse.BodyHandlers.discarding());
HttpResponse<Void> response = httpClient.send(request, HttpResponse.BodyHandlers.discarding());
if (response.statusCode() != 201) {
logger.error("Unexpected status code: {}", response.statusCode());
}
}
public Optional<String> getMessage(int channelId, int messageId) throws IOException, InterruptedException {
final var uri = URI.create(DISCORD_URL + "/channels/%s/messages/%s".formatted(channelId, messageId));
final HttpRequest request = HttpRequest.newBuilder()
var uri = URI.create(DISCORD_URL + "/channels/%s/messages/%s".formatted(channelId, messageId));
HttpRequest request = HttpRequest.newBuilder()
.uri(uri)
.header("Authorization", "Bot " + webConfig.token())
.header("User-Agent", USER_AGENT)
.GET()
.build();
final HttpResponse<String> response = httpClient.send(request, HttpResponse.BodyHandlers.ofString());
HttpResponse<String> response = httpClient.send(request, HttpResponse.BodyHandlers.ofString());
if (response.statusCode() != 200) {
logger.error("Could not get message");
return Optional.empty();

View file

@ -8,6 +8,7 @@ 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;
@ -29,10 +30,8 @@ public final class DiscordWebSocket {
instance.start();
}
public void start() throws InterruptedException {
public void start() {
final var connector = new Connector(eventHandler);
final String initGatewayUrl = "wss://gateway.discord.gg/?v=10&encoding=json";
final var connectionManager = new ConnectionManager(initGatewayUrl, discordApiKey, retryer);
connectionManager.start(connector);
final var connectionManager = new ConnectionManager(DISCORD_API_URL, discordApiKey, retryer);
}
}

View file

@ -10,7 +10,7 @@ public final class IdendificationConnectionInitiator implements ConnectionInitia
}
@Override
public void initiate(final WebSocket webSocket) {
public void initiate(WebSocket webSocket) {
final String identifyPayload = "{\"op\": 2, \"d\": {\"token\": \"" + token + "\", \"intents\": 513, \"properties\": {\"os\": \"linux\", \"browser\": \"dchat_lib\", \"device\": \"dchat_lib\"}}}";
webSocket.sendText(identifyPayload, true);
}

View file

@ -14,7 +14,7 @@ public final class ResumeConnectionInitiator implements ConnectionInitiator {
}
@Override
public void initiate(final WebSocket webSocket) {
public void initiate(WebSocket webSocket) {
final String identifyPayload = "{\"op\": 6, \"d\": {\"token\": \"" + token + "\", \"session_id\": \"" + sessionId + "\", \"seq\": " + lastSequence + "}}";
webSocket.sendText(identifyPayload, true);
}

View file

@ -6,36 +6,6 @@ import org.json.JSONObject;
import java.util.Map;
public final class EventDeserializer {
private static EventType resolveEventType(final String type) {
if (type.isBlank()) return new EventType.Empty();
if ("READY".equals(type)) return new EventType.Ready();
return new EventType.Unknown(type);
}
private static int extractPossibleNullInt(final JSONObject jsonObject, final String key) {
try {
return jsonObject.getInt(key);
} catch (JSONException e) {
return 0;
}
}
private static String extractPossibleNullString(final JSONObject jsonObject, final String key) {
try {
return jsonObject.getString(key);
} catch (JSONException e) {
return "";
}
}
private static JSONObject extractPossibleNullObject(final JSONObject jsonObject, final String key) {
try {
return jsonObject.getJSONObject(key);
} catch (JSONException e) {
return new JSONObject(Map.of());
}
}
public Event deserialize(final String text) {
final JSONObject object = new JSONObject(text);
final String type = extractPossibleNullString(object, "t");
@ -45,4 +15,34 @@ public final class EventDeserializer {
final JSONObject data = extractPossibleNullObject(object, "d");
return new Event(eventType, sequence, operation, data);
}
private EventType resolveEventType(final String type) {
if (type.isBlank()) return new EventType.Empty();
if ("READY".equals(type)) return new EventType.Ready();
return new EventType.Unknown(type);
}
private int extractPossibleNullInt(final JSONObject jsonObject, final String key) {
try {
return jsonObject.getInt(key);
} catch (JSONException e) {
return 0;
}
}
private String extractPossibleNullString(final JSONObject jsonObject, final String key) {
try {
return jsonObject.getString(key);
} catch (JSONException e) {
return "";
}
}
private JSONObject extractPossibleNullObject(final JSONObject jsonObject, final String key) {
try {
return jsonObject.getJSONObject(key);
} catch (JSONException e) {
return new JSONObject(Map.of());
}
}
}

View file

@ -2,6 +2,7 @@ package de.hhhammer.dchat.openai.rest;
import de.hhhammer.dchat.openai.rest.MessageContext.PreviousInteraction;
import de.hhhammer.dchat.openai.rest.MessageContext.ReplyInteraction;
import de.hhhammer.dchat.openai.rest.models.ChatGPTRequest;
import org.jetbrains.annotations.NotNull;
import java.util.ArrayList;

View file

@ -1,22 +1,15 @@
package de.hhhammer.dchat.openai.rest;
import org.jetbrains.annotations.Nullable;
import org.json.JSONArray;
import org.json.JSONException;
import org.json.JSONObject;
import de.hhhammer.dchat.openai.rest.models.ChatGPTRequest;
import de.hhhammer.dchat.openai.rest.models.ChatGPTResponse;
import java.io.IOException;
import java.io.InputStream;
import java.net.URI;
import java.net.http.HttpClient;
import java.net.http.HttpRequest;
import java.net.http.HttpResponse;
import java.time.Duration;
import java.util.Objects;
import java.util.Optional;
import java.util.Spliterator;
import java.util.Spliterators;
import java.util.function.Supplier;
import java.util.stream.StreamSupport;
public final class ChatGPTService {
private final String apiKey;
@ -27,15 +20,7 @@ public final class ChatGPTService {
this.httpClient = httpClient;
}
private static <T> @Nullable T getFromJson(final Supplier<T> supplier) {
try {
return supplier.get();
} catch (final JSONException e) {
return null;
}
}
public Optional<String> submit(final ChatGPTRequest chatGPTRequest) throws IOException, InterruptedException, ResponseException {
public ChatGPTResponse submit(final ChatGPTRequest chatGPTRequest) throws IOException, InterruptedException, ResponseException {
final String data = chatGPTRequest.toJson().toString();
final URI uri = URI.create("https://api.openai.com/v1/chat/completions");
final HttpRequest request = HttpRequest.newBuilder(uri)
@ -45,21 +30,11 @@ public final class ChatGPTService {
.timeout(Duration.ofMinutes(5))
.build();
final HttpResponse<String> response = httpClient.send(request, HttpResponse.BodyHandlers.ofString());
if (response.statusCode() != 200) {
throw new ResponseException("Response status code was not 200: " + response.statusCode());
final HttpResponse<InputStream> responseStream = httpClient.send(request, HttpResponse.BodyHandlers.ofInputStream());
if (responseStream.statusCode() != 200) {
throw new ResponseException("Response status code was not 200: " + responseStream.statusCode());
}
final var responseJson = new JSONObject(response);
final JSONArray choices = responseJson.getJSONArray("choices");
final Spliterator<Object> spliterator = Spliterators.spliterator(choices.iterator(), choices.length(), Spliterator.SIZED);
return StreamSupport.stream(spliterator, false)
.filter(JSONObject.class::isInstance)
.map(JSONObject.class::cast)
.map(choice -> getFromJson(() -> choice.getJSONObject("message")))
.filter(Objects::nonNull)
.map(message -> getFromJson(() -> message.getString("content")))
.filter(Objects::nonNull)
.findFirst();
return mapper.readValue(responseStream.body(), ChatGPTResponse.class);
}
}

View file

@ -1,5 +1,6 @@
package de.hhhammer.dchat.openai.rest;
package de.hhhammer.dchat.openai.rest.models;
import de.hhhammer.dchat.openai.rest.JsonSerializable;
import org.jetbrains.annotations.NotNull;
import org.json.JSONArray;
import org.json.JSONObject;

View file

@ -0,0 +1,49 @@
package de.hhhammer.dchat.openai.rest.models;
import de.hhhammer.dchat.openai.rest.JsonSerializable;
import org.jetbrains.annotations.NotNull;
import org.json.JSONArray;
import org.json.JSONObject;
import java.util.List;
public record ChatGPTResponse(Usage usage, List<Choice> choices) implements JsonSerializable {
@Override
public @NotNull JSONObject toJson() {
final var json = new JSONObject();
json.put("usage", usage.toJson());
final var jsonChoices = new JSONArray(choices.stream().map(Choice::toJson).toList());
json.put("choices", jsonChoices);
return json;
}
public record Usage(int promptTokens, int completionTokens, int totalTokens) implements JsonSerializable {
@Override
public @NotNull JSONObject toJson() {
final var json = new JSONObject();
json.put("prompt_tokens", promptTokens);
json.put("completion_tokens", completionTokens);
json.put("total_tokens", totalTokens);
return json;
}
}
public record Choice(Message message) implements JsonSerializable {
@Override
public @NotNull JSONObject toJson() {
final var json = new JSONObject();
json.put("message", message.toJson());
return json;
}
public record Message(String role, String content) implements JsonSerializable {
@Override
public @NotNull JSONObject toJson() {
final var json = new JSONObject();
json.put("role", role);
json.put("content", content);
return json;
}
}
}
}