Skip to content

Latest commit

 

History

History
363 lines (299 loc) · 15 KB

Using_RestAction.md

File metadata and controls

363 lines (299 loc) · 15 KB

This wiki page is migrating to jda.wiki/using-jda/using-restaction


What is a RestAction?

If you understand RestAction you understand JDA.

In JDA 3.0 we introduced the new RestAction class which basically is a terminal between the JDA user and the Discord REST API.
The RestAction is a step between specifying what the user wants to do and executing it, it allows the user to specify how JDA should deal with their Request.

However this only works if you actually tell the RestAction to do something. That is why we recommend checking out whether or not something in JDA returns a RestAction. If that is the case you have to execute it using one of the RestAction execution operations:

  • queue(), queue(Consumer), queue(Consumer, Consumer)
    These operations are asynchronous and will not execute within the same Thread.
    This means that you cannot use procedural logic when you use queue(), unless you use the callback Consumers.
    Only similar requests are internally executed in sequence such as sending messages in the same channel or adding reactions to the same message.
  • submit()
    Provides request future to cancel tasks later and avoid callback hell.
  • complete()
    This operation will block the current Thread until the request has been finished and will return the response type.

Note: We recommend using queue() or submit() when possible as blocking the current Thread can cause downtime and will use more resources.

Since 4.1.1 you can use a few RestAction operators to avoid callback hell with queue:

  • map Convert the result of the RestAction to a different value
  • flatMap Chain another RestAction on the result
  • delay Delay the element of the previous step

JavaDocs: https://ci.dv8tion.net/job/JDA/javadoc/net/dv8tion/jda/api/requests/RestAction.html

AuditLog Reasons

Some operations return a special RestAction implementation called AuditableRestAction.
This extension allows to set a reason field for that action.

These reasons are only available for accounts with AccountType.BOT but will silently be ignored for any non-bot accounts.
However any account can use reasons on kick/ban with the special overloads provided that allow setting a reason.

Example:

public class ModerationUtil
{
    public static void deleteMessage(Message message, String reason)
    {
        message.delete().reason(reason).queue();
    }

    public static void ban(Guild guild, User user, String reason)
    {
        guild.ban(user, 7, reason).queue();
    }
}

Using queue()

The most common way to execute a RestAction is by simply calling .queue() after the operation:

public void sendMessage(MessageChannel channel, String message) 
{
    channel.sendMessage(message).queue();
}

This will always simply execute the RestAction<Message> which was returned by MessageChannel.sendMessage(String).
Note that this might happen after calling sendMessage(MessageChannel, String) because queue() is asynchronous!

You: Why can't I access the Message that was sent with queue()?
Minn: Use the success callback!

Is one of the common conversations we had when people started using JDA 3.0.

You: What does that mean?

A success callback is what we call the primary Consumer that can be passed to a queue() statement:

public void sendAndLog(MessageChannel channel, String message) 
{
    channel.sendMessage(message).queue(new Consumer<Message>()
    {
        @Override
        public void accept(Message t)
        {
            System.out.printf("Sent Message %s\n", t);
        }
    });
}

Here we used an inline implementation of Consumer<Message> that handles the response of a REST Request.
The method Consumer.accept(Message) is automatically called once the response has been received by the JDA Requester.

Minn: But that looks really ugly...
You: Yeah but it works!!

Since JDA requires you to use Java 1.8 we can use one of the new features: Lambda Expressions

public void sendAndLog(MessageChannel channel, String message)
{
    // Here we use a lambda expressions which names the callback parameter -response- and uses that as a reference
    // in the callback body -System.out.printf("Sent Message %s\n", response)-
    Consumer<Message> callback = (response) -> System.out.printf("Sent Message %s\n", response);
    channel.sendMessage(message).queue(callback); // ^ calls that
}

You: Wow that looks so much better!
Minn: Yes, please learn more about lambda expressions: lambda quickstart

Example: Sending a Private Message

public void sendPrivateMessage(User user, String content)
{
    // openPrivateChannel provides a RestAction<PrivateChannel> 
    // which means it supplies you with the resulting channel
    user.openPrivateChannel().queue((channel) ->
    {
        // value is a parameter for the `accept(T channel)` method of our callback.
        // here we implement the body of that method, which will be called later by JDA automatically.
        channel.sendMessage(content).queue();
        // here we access the enclosing scope variable -content-
        // which was provided to sendPrivateMessage(User, String) as a parameter
    });
}

Since this only calls a single method in the callback you can use the short form:

public void sendPrivateMessage(User user, String content)
{
    // notice that we are not placing a semicolon (;) in the callback this time!
    user.openPrivateChannel().queue( (channel) -> channel.sendMessage(content).queue() );
}

Using submit()

Sometimes execution needs to be cancelled if it isn't required anymore. This can be challenging to do if you use queue() or complete().
In submit() JDA will provide a CompletableFuture (aka promise) which allows the cancellation of a request.

If you don't need to use the CompletableFuture you may use queue() instead!

Example

public void setTestingChannel(TextChannel channel)
{
    channel.getManager().setName("testing-channel").queue( (v) ->
        channel.sendMessage("Update Channel").queue( (m) ->
            m.delete().queueAfter(30, TimeUnit.SECONDS, (t) ->
                logChannel.sendMessage("Deleted Response in %s", channel).queue()
            )
        )
    );
}

// turns into
public void setTestingChannel(TextChannel channel)
{
    channel.getManager().setName("testing-channel").submit()                   // CompletableFuture<Void>
           .thenCompose((v) -> channel.sendMessage("Update Channel").submit()) // CompletableFuture<Message>
           .thenCompose((m) -> m.delete().submitAfter(30, TimeUnit.SECONDS))   // CompletableFuture<Void>
           .thenCompose((v) -> logChannel.sendMessage("Deleted Response in %s", channel).submit())
           .whenComplete((s, error) -> {
               // this will be called for every termination (success/failure)
               // if the result is successful the error will be null
               // otherwise you should handle the error here to prevent it from being eaten and never printed
               if (error != null) error.printStackTrace();
           });
}

Note: You can do the same with RestAction#flatMap in 4.1.1

public class RateLimitListener extends ListenerAdapter
{
    private final long guildId;
    private final long userId;
    private final Queue<RequestFuture<Void>> tasks = new LinkedList<>();

    public RateLimitListener(Guild guild, User user)
    {
        guildId = guild.getIdLong();
        userId = user.getIdLong();
        // only store IDs as JDA objects can be disposed by cache invalidation
        //when disposed the entity is not usable anymore, since we only need the id this is good enough
    }

    @Override
    public void onGuildMessageReceived(GuildMessageReceivedEvent event)
    {
        if (event.getAuthor().getIdLong() != userId)
            return; // ignore other users
        if (event.getGuild().getIdLong() != guildId)
            return; // ignore other guilds
        RequestFuture<Void> task = event.getMessage().delete().submit();
        tasks.add(task); // add task to cancel queue in case user gets banned
        task.thenRun(() -> tasks.remove(task)); // remove once completed
    }

    @Override
    public void onGuildBan(GuildBanEvent event)
    {
        if (event.getUser().getIdLong() != userId)
            return; // ignore other users
        if (event.getGuild().getIdLong() != guildId)
            return; // ignore other guilds

        // stop deleting messages for banned user
        RequestFuture<Void> current;
        while ((current = tasks.poll()) != null)
            current.cancel(true); 
        tasks.clear();

        // remove this as listener, our task has completed!
        event.getJDA().removeEventListener(this);
    }
}

Using complete()

The complete() operation is simply for your convenience. It will block the Thread that you call it on which means it will not be able to continue with other tasks in the meantime.
If you don't use the return value or don't need the request to be completed before continuing with other operations it is recommended to use queue() instead!

Examples

public void setTestingChannel(TextChannel channel)
{
    channel.getManager().setName("testing-channel").queue( (v) ->
        channel.sendMessage("Update Channel").queue( (m) ->
            m.delete().queueAfter(30, TimeUnit.SECONDS, (t) ->
                logChannel.sendMessage("Deleted Response in %s", channel).queue()
            )
        )
    );
}

public void setTestingChannelBlocking(TextChannel channel)
{
    channel.getManager().setName("testing-channel").complete();
    Message m = channel.sendMessage("Update Channel").complete();
    m.delete().completeAfter(30, TimeUnit.SECONDS);
    logChannel.sendMessage("Deleted Response in %s", channel).queue();
    // note how we used queue in the end because we don't need it sequenced anymore.
}

This is called a callback hell

public Message sendAndLog(MessageChannel channel, String message)
{
    Message response = channel.sendMessage(message).complete();
    System.out.printf("Sent Message %s\n", response);
    return response;
}
public PermissionOverride getOverride(Channel channel, Member member)
{
    final PermissionOverride override = channel.getPermissionOverride(member);
    
    if (override == null)
        return channel.createPermissionOverride(member).complete();
    
    return override;
}

You can do this asynchronously by using a CompletableFuture:

public CompletableFuture<PermissionOverride> getOverride(Channel channel, Member member)
{
    final PermissionOverride override = channel.getPermissionOverride(member);
    
    if (override == null)
        return channel.createPermissionOverride(member).submit();
    
    return CompletableFuture.completedFuture(override);
}

getOverride(channel, member).thenAccept(override -> ...);

Using completeAfter, submitAfter and queueAfter

These three methods are also known under the term of Planned Execution as they use a ScheduledExecutorService to schedule calls to either [[complete|7)-Using-RestAction#using-complete]] or [[queue|7)-Using-RestAction#using-queue]].

There are three possible ways to plan a RestAction execution

  • [[completeAfter(long, TimeUnit)|7)-Using-RestAction#example-completeafter]]
    Blocks and executes on the current Thread, similar to complete()! Similar to using Thread.join() this will block until the action has completed.
  • [[submitAfter(long, TimeUnit), submitAfter(long, TimeUnit, ScheduledExecutorService)|7)-Using-RestAction#example-submitafter]]
    Creates a DelayedCompletableFuture<T> which will hold the response type as its generic value. This means using get() on the returned Future will cause the current thread to block and await the execution of the RestAction and receive the response type.
  • [[queueAfter(long, TimeUnit), queueAfter(long, TimeUnit, Consumer<T>), queueAfter(long, TimeUnit, Consumer<T>, Consumer<Throwable>)|7)-Using-RestAction#example-queueafter]]
    Schedules the RestAction execution to be started after the specified delay, this will not block the thread and handle the execution in the background. You can optionally provide a ScheduledExecutorService to any of the queueAfter operations as the last argument.

When no ScheduledExecutorService is provided, these operations will use the default internal JDA ScheduledExecutorService that is also used to execute queue callback consumers.

Example completeAfter

public Message waitForEdit(Message message)
{
    return message.editMessage("5 Minutes are over").completeAfter(5, TimeUnit.MINUTES);
}

Example queueAfter

public void remind(User user, String reminder, long delay, TimeUnit unit)
{
    user.openPrivateChannel().queue(
        (channel) -> channel.sendMessage(reminder).queueAfter(delay, unit)
    );
}

public void remindAlternate(User user, String reminder, long delay, TimeUnit unit)
{
    user.openPrivateChannel().queueAfter(delay, unit,
        (channel) -> channel.sendMessage(reminder).queue()
    );
}

Example submitAfter

private Map<String, DelayedCompletableFuture<Message>> tasks = new HashMap<>();

public ScheduledFuture<Message> sendWithTask(MessageChannel channel, String message)
{
    DelayedCompletable<Message> task = channel.sendMessage(message).submitAfter(5, TimeUnit.SECONDS);
    return task;
}

public void doSomething(MessageChannel channel, String message)
throws Exception
{
    tasks.add(channel.getId(), sendWithTask(channel, message));
    for (DelayedCompletable<Message> task : tasks.values())
    {
        // non-blocking alternative is `thenAccept`
        System.out.printf("Task completed: %s\n", task.get());
    }
}