-
-
Notifications
You must be signed in to change notification settings - Fork 90
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
1158: Add boilerplate for the slash command
- Loading branch information
1 parent
12c42a5
commit 81657be
Showing
10 changed files
with
349 additions
and
1 deletion.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
168 changes: 168 additions & 0 deletions
168
application/src/main/java/org/togetherjava/tjbot/features/jshell/aws/JShellAWSCommand.java
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,168 @@ | ||
package org.togetherjava.tjbot.features.jshell.aws; | ||
|
||
import com.fasterxml.jackson.core.JsonProcessingException; | ||
import com.fasterxml.jackson.databind.ObjectMapper; | ||
import net.dv8tion.jda.api.EmbedBuilder; | ||
import net.dv8tion.jda.api.entities.Member; | ||
import net.dv8tion.jda.api.events.interaction.command.SlashCommandInteractionEvent; | ||
import net.dv8tion.jda.api.interactions.InteractionHook; | ||
import net.dv8tion.jda.api.interactions.commands.OptionMapping; | ||
import org.apache.logging.log4j.LogManager; | ||
import org.apache.logging.log4j.Logger; | ||
|
||
import org.togetherjava.tjbot.features.CommandVisibility; | ||
import org.togetherjava.tjbot.features.SlashCommandAdapter; | ||
import org.togetherjava.tjbot.features.jshell.aws.exceptions.JShellAPIException; | ||
|
||
import java.awt.*; | ||
|
||
public class JShellAWSCommand extends SlashCommandAdapter { | ||
private static final Logger logger = LogManager.getLogger(JShellAWSCommand.class); | ||
private static final ObjectMapper OBJECT_MAPPER = new ObjectMapper(); | ||
private static final String CODE_PARAMETER = "code"; | ||
private final JShellService jShellService; | ||
|
||
public JShellAWSCommand(JShellService jShellService) { | ||
super("jshell-aws", "Execute Java code in Discord!", CommandVisibility.GUILD); | ||
this.jShellService = jShellService; | ||
} | ||
|
||
@Override | ||
public void onSlashCommand(SlashCommandInteractionEvent event) { | ||
Member member = event.getMember(); | ||
|
||
if (member == null) { | ||
event.reply("Member that executed the command is no longer available, won't execute") | ||
.queue(); | ||
return; | ||
} | ||
|
||
OptionMapping input = event.getOption(CODE_PARAMETER); | ||
|
||
if (input == null) { | ||
event.reply("Please provide a value for the code parameter").queue(); | ||
return; | ||
} | ||
|
||
event.deferReply().queue(); | ||
|
||
InteractionHook hook = event.getHook(); | ||
|
||
try { | ||
respondWithJShellOutput(hook, member, | ||
jShellService.sendRequest(new JShellRequest(input.getAsString()))); | ||
} catch (JShellAPIException jShellAPIException) { | ||
if (jShellAPIException.getStatusCode() == 400) { | ||
logger.warn("HTTP 400 error occurred with the JShell AWS API {}", | ||
jShellAPIException.getBody()); | ||
respondWithInputError(hook, jShellAPIException.getBody()); | ||
} else if (jShellAPIException.getStatusCode() == 408) { | ||
respondWithTimeout(hook); | ||
} else { | ||
logger.error("HTTP {} received from JShell AWS API {}", | ||
jShellAPIException.getStatusCode(), jShellAPIException.getBody()); | ||
respondWithSevereAPIError(hook); | ||
} | ||
} catch (Exception e) { | ||
logger.error( | ||
"An error occurred while sending/receiving request from the AWS JShell API", e); | ||
respondWithSevereAPIError(hook); | ||
} | ||
} | ||
|
||
private static void respondWithJShellOutput(InteractionHook hook, Member member, | ||
JShellResponse response) { | ||
// Extracted as fields to be compliant with Sonar | ||
final String SNIPPET_SECTION_TITLE = "## Snippets\n"; | ||
final String BACKTICK = "`"; | ||
final String NEWLINE = "\n"; | ||
final String DOUBLE_NEWLINE = "\n\n"; | ||
final String STATUS = "**Status**: "; | ||
final String OUTPUT_SECTION_TITLE = "**Output**\n"; | ||
final String JAVA_CODE_BLOCK_START = "```java\n"; | ||
final String CODE_BLOCK_END = "```\n"; | ||
final String DIAGNOSTICS_SECTION_TITLE = "**Diagnostics**\n"; | ||
final String CONSOLE_OUTPUT_SECTION_TITLE = "## Console Output\n"; | ||
final String ERROR_OUTPUT_SECTION_TITLE = "## Error Output\n"; | ||
|
||
StringBuilder sb = new StringBuilder(); | ||
sb.append(SNIPPET_SECTION_TITLE); | ||
|
||
for (JShellSnippet snippet : response.events()) { | ||
sb.append(BACKTICK); | ||
sb.append(snippet.statement()); | ||
sb.append(BACKTICK).append(DOUBLE_NEWLINE); | ||
sb.append(STATUS); | ||
sb.append(snippet.status()); | ||
sb.append(NEWLINE); | ||
|
||
if (snippet.value() != null && !snippet.value().isEmpty()) { | ||
sb.append(OUTPUT_SECTION_TITLE); | ||
sb.append(JAVA_CODE_BLOCK_START); | ||
sb.append(snippet.value()); | ||
sb.append(CODE_BLOCK_END); | ||
} | ||
|
||
if (!snippet.diagnostics().isEmpty()) { | ||
sb.append(DIAGNOSTICS_SECTION_TITLE); | ||
for (String diagnostic : snippet.diagnostics()) { | ||
sb.append(BACKTICK).append(diagnostic).append(BACKTICK).append(NEWLINE); | ||
} | ||
} | ||
} | ||
|
||
if (response.outputStream() != null && !response.outputStream().isEmpty()) { | ||
sb.append(CONSOLE_OUTPUT_SECTION_TITLE); | ||
sb.append(JAVA_CODE_BLOCK_START); | ||
sb.append(response.outputStream()); | ||
sb.append(CODE_BLOCK_END); | ||
} | ||
|
||
if (response.errorStream() != null && !response.errorStream().isEmpty()) { | ||
sb.append(ERROR_OUTPUT_SECTION_TITLE); | ||
sb.append(JAVA_CODE_BLOCK_START); | ||
sb.append(response.errorStream()); | ||
sb.append(CODE_BLOCK_END); | ||
} | ||
|
||
EmbedBuilder eb = new EmbedBuilder(); | ||
eb.setAuthor(member.getEffectiveName(), null, member.getAvatarUrl()); | ||
if (sb.length() > 4000) { | ||
eb.setDescription(sb.substring(0, 4000)); | ||
} else { | ||
eb.setDescription(sb.toString()); | ||
} | ||
eb.setColor(Color.GREEN); | ||
|
||
hook.editOriginalEmbeds(eb.build()).queue(); | ||
} | ||
|
||
|
||
private static void respondWithInputError(InteractionHook hook, String response) { | ||
JShellErrorResponse errorResponse; | ||
try { | ||
errorResponse = OBJECT_MAPPER.readValue(response, JShellErrorResponse.class); | ||
} catch (JsonProcessingException e) { | ||
errorResponse = new JShellErrorResponse( | ||
"There was a problem with the input you provided, please check and try again"); | ||
} | ||
EmbedBuilder eb = new EmbedBuilder(); | ||
eb.setDescription(errorResponse.error()); | ||
eb.setColor(Color.ORANGE); | ||
hook.editOriginalEmbeds(eb.build()).queue(); | ||
} | ||
|
||
private static void respondWithTimeout(InteractionHook hook) { | ||
EmbedBuilder eb = new EmbedBuilder(); | ||
eb.setDescription("The code you provided took too long and the request has timed out!"); | ||
eb.setColor(Color.ORANGE); | ||
hook.editOriginalEmbeds(eb.build()).queue(); | ||
} | ||
|
||
private static void respondWithSevereAPIError(InteractionHook hook) { | ||
EmbedBuilder eb = new EmbedBuilder(); | ||
eb.setDescription("An internal error occurred, please try again later"); | ||
eb.setColor(Color.RED); | ||
hook.editOriginalEmbeds(eb.build()).queue(); | ||
} | ||
} |
13 changes: 13 additions & 0 deletions
13
...ication/src/main/java/org/togetherjava/tjbot/features/jshell/aws/JShellErrorResponse.java
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,13 @@ | ||
package org.togetherjava.tjbot.features.jshell.aws; | ||
|
||
import com.fasterxml.jackson.annotation.JsonIgnoreProperties; | ||
import com.fasterxml.jackson.annotation.JsonProperty; | ||
|
||
/** | ||
* Represents a response from JShell that contains an error key. | ||
* | ||
* @author Suraj Kuamr | ||
*/ | ||
@JsonIgnoreProperties(ignoreUnknown = true) | ||
public record JShellErrorResponse(@JsonProperty("error") String error) { | ||
} |
11 changes: 11 additions & 0 deletions
11
application/src/main/java/org/togetherjava/tjbot/features/jshell/aws/JShellRequest.java
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,11 @@ | ||
package org.togetherjava.tjbot.features.jshell.aws; | ||
|
||
/** | ||
* A record containing the code snippet to be evaluated by the AWS JShell API | ||
* | ||
* @param code The Java code snippet to execute | ||
* | ||
* @author Suraj Kumar | ||
*/ | ||
public record JShellRequest(String code) { | ||
} |
21 changes: 21 additions & 0 deletions
21
application/src/main/java/org/togetherjava/tjbot/features/jshell/aws/JShellResponse.java
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,21 @@ | ||
package org.togetherjava.tjbot.features.jshell.aws; | ||
|
||
import com.fasterxml.jackson.annotation.JsonIgnoreProperties; | ||
import com.fasterxml.jackson.annotation.JsonProperty; | ||
|
||
import java.util.List; | ||
|
||
/** | ||
* A record containing the AWS JShell API response. | ||
* | ||
* @param errorStream The content in JShells error stream | ||
* @param outputStream The content in JShells standard output stream | ||
* @param events A list of snippets that were evaluated | ||
* | ||
* @author Suraj Kumar | ||
*/ | ||
@JsonIgnoreProperties(ignoreUnknown = true) | ||
public record JShellResponse(@JsonProperty("errorStream") String errorStream, | ||
@JsonProperty("outputStream") String outputStream, | ||
@JsonProperty("events") List<JShellSnippet> events) { | ||
} |
76 changes: 76 additions & 0 deletions
76
application/src/main/java/org/togetherjava/tjbot/features/jshell/aws/JShellService.java
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,76 @@ | ||
package org.togetherjava.tjbot.features.jshell.aws; | ||
|
||
import com.fasterxml.jackson.core.JsonProcessingException; | ||
import com.fasterxml.jackson.databind.ObjectMapper; | ||
import org.apache.logging.log4j.LogManager; | ||
import org.apache.logging.log4j.Logger; | ||
|
||
import org.togetherjava.tjbot.features.jshell.aws.exceptions.JShellAPIException; | ||
|
||
import java.io.IOException; | ||
import java.net.URI; | ||
import java.net.URISyntaxException; | ||
import java.net.http.HttpClient; | ||
import java.net.http.HttpRequest; | ||
import java.net.http.HttpResponse; | ||
|
||
/** | ||
* The JShellService class is used to interact with the AWS JShell API. | ||
* | ||
* @author Suraj Kumar | ||
*/ | ||
public class JShellService { | ||
private static final Logger LOGGER = LogManager.getLogger(JShellService.class); | ||
private static final ObjectMapper OBJECT_MAPPER = new ObjectMapper(); | ||
private final String apiURL; | ||
private final HttpClient httpClient; | ||
|
||
/** | ||
* Constructs a JShellService. | ||
* | ||
* @param apiURl The Lambda Function URL to send API requests to | ||
*/ | ||
public JShellService(String apiURl) { | ||
this.apiURL = apiURl; | ||
this.httpClient = HttpClient.newHttpClient(); | ||
} | ||
|
||
/** | ||
* Sends an HTTP request to the AWS JShell API. | ||
* | ||
* @param jShellRequest The request object containing the code to evaluate | ||
* @return The API response as a JShellResponse object | ||
* @throws URISyntaxException If the API URL is invalid | ||
* @throws JsonProcessingException If the API response failed to get parsed by Jackson to our | ||
* mapping. | ||
*/ | ||
public JShellResponse sendRequest(JShellRequest jShellRequest) | ||
throws URISyntaxException, JsonProcessingException { | ||
HttpRequest request = HttpRequest.newBuilder() | ||
.uri(new URI(apiURL)) | ||
.header("Content-Type", "application/json") | ||
.POST(HttpRequest.BodyPublishers | ||
.ofString(OBJECT_MAPPER.writeValueAsString(jShellRequest))) | ||
.build(); | ||
|
||
try { | ||
HttpResponse<String> response = | ||
httpClient.send(request, HttpResponse.BodyHandlers.ofString()); | ||
|
||
if (response.statusCode() != 200) { | ||
throw new JShellAPIException(response.statusCode(), response.body()); | ||
} | ||
|
||
String body = response.body(); | ||
LOGGER.trace("Received the following body from the AWS JShell API: {}", body); | ||
|
||
return OBJECT_MAPPER.readValue(response.body(), JShellResponse.class); | ||
|
||
} catch (IOException | InterruptedException e) { | ||
LOGGER.error("Failed to send http request to the AWS JShell API", e); | ||
Thread.currentThread().interrupt(); | ||
} | ||
|
||
return null; | ||
} | ||
} |
23 changes: 23 additions & 0 deletions
23
application/src/main/java/org/togetherjava/tjbot/features/jshell/aws/JShellSnippet.java
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,23 @@ | ||
package org.togetherjava.tjbot.features.jshell.aws; | ||
|
||
import com.fasterxml.jackson.annotation.JsonIgnoreProperties; | ||
import com.fasterxml.jackson.annotation.JsonProperty; | ||
|
||
import java.util.List; | ||
|
||
/** | ||
* A JShell snippet is a statement that is to be executed. This record is used to hold information | ||
* about a statement that was provided by the AWS JShell API | ||
* | ||
* @param statement The statement that was executed | ||
* @param value The return value of the statement | ||
* @param status The status from evaluating the statement e.g. "VALID", "INVALID" | ||
* @param diagnostics A list of diagnostics such as error messages provided by JShell | ||
* | ||
* @author Suraj Kumar | ||
*/ | ||
@JsonIgnoreProperties(ignoreUnknown = true) | ||
public record JShellSnippet(@JsonProperty("statement") String statement, | ||
@JsonProperty("value") String value, @JsonProperty("status") String status, | ||
List<String> diagnostics) { | ||
} |
Oops, something went wrong.