Merge pull request #835 from Chase22/ability-reply-inject-bot

Inject bot instance into the reply update consumer
This commit is contained in:
Ruben Bermudez 2021-03-06 11:15:11 +00:00 committed by GitHub
commit f724181daf
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
8 changed files with 95 additions and 30 deletions

View File

@ -625,7 +625,7 @@ public abstract class BaseAbilityBot extends DefaultAbsSender implements Ability
return replies.stream() return replies.stream()
.filter(reply -> runSilently(() -> reply.isOkFor(update), reply.name())) .filter(reply -> runSilently(() -> reply.isOkFor(update), reply.name()))
.map(reply -> runSilently(() -> { .map(reply -> runSilently(() -> {
reply.actOn(update); reply.actOn(this, update);
updateReplyStats(reply); updateReplyStats(reply);
return false; return false;
}, reply.name())) }, reply.name()))

View File

@ -254,7 +254,7 @@ public final class DefaultAbilities implements AbilityExtension {
.input(0) .input(0)
.action(ctx -> bot.silent.forceReply( .action(ctx -> bot.silent.forceReply(
getLocalizedMessage(ABILITY_RECOVER_MESSAGE, ctx.user().getLanguageCode()), ctx.chatId())) getLocalizedMessage(ABILITY_RECOVER_MESSAGE, ctx.user().getLanguageCode()), ctx.chatId()))
.reply(update -> { .reply((bot, update) -> {
String replyToMsg = update.getMessage().getReplyToMessage().getText(); String replyToMsg = update.getMessage().getReplyToMessage().getText();
String recoverMessage = getLocalizedMessage(ABILITY_RECOVER_MESSAGE, AbilityUtils.getUser(update).getLanguageCode()); String recoverMessage = getLocalizedMessage(ABILITY_RECOVER_MESSAGE, AbilityUtils.getUser(update).getLanguageCode());
if (!replyToMsg.equals(recoverMessage)) if (!replyToMsg.equals(recoverMessage))

View File

@ -4,10 +4,12 @@ import com.google.common.base.MoreObjects;
import com.google.common.base.Objects; import com.google.common.base.Objects;
import org.slf4j.Logger; import org.slf4j.Logger;
import org.slf4j.LoggerFactory; import org.slf4j.LoggerFactory;
import org.telegram.abilitybots.api.bot.BaseAbilityBot;
import org.telegram.telegrambots.meta.api.objects.Update; import org.telegram.telegrambots.meta.api.objects.Update;
import java.util.Arrays; import java.util.Arrays;
import java.util.List; import java.util.List;
import java.util.function.BiConsumer;
import java.util.function.Consumer; import java.util.function.Consumer;
import java.util.function.Predicate; import java.util.function.Predicate;
@ -208,12 +210,22 @@ public final class Ability {
return this; return this;
} }
/**
* @deprecated Please use {@link #reply(BiConsumer, Predicate[])}
*/
@Deprecated
@SafeVarargs @SafeVarargs
public final AbilityBuilder reply(Consumer<Update> action, Predicate<Update>... conditions) { public final AbilityBuilder reply(Consumer<Update> action, Predicate<Update>... conditions) {
replies.add(Reply.of(action, conditions)); replies.add(Reply.of(action, conditions));
return this; return this;
} }
@SafeVarargs
public final AbilityBuilder reply(BiConsumer<BaseAbilityBot, Update> action, Predicate<Update>... conditions) {
replies.add(Reply.of(action, conditions));
return this;
}
public final AbilityBuilder reply(Reply reply) { public final AbilityBuilder reply(Reply reply) {
replies.add(reply); replies.add(reply);
return this; return this;

View File

@ -2,10 +2,12 @@ package org.telegram.abilitybots.api.objects;
import com.google.common.base.MoreObjects; import com.google.common.base.MoreObjects;
import com.google.common.collect.ImmutableList; import com.google.common.collect.ImmutableList;
import org.telegram.abilitybots.api.bot.BaseAbilityBot;
import org.telegram.telegrambots.meta.api.objects.Update; import org.telegram.telegrambots.meta.api.objects.Update;
import java.util.List; import java.util.List;
import java.util.Objects; import java.util.Objects;
import java.util.function.BiConsumer;
import java.util.function.BiFunction; import java.util.function.BiFunction;
import java.util.function.Consumer; import java.util.function.Consumer;
import java.util.function.Predicate; import java.util.function.Predicate;
@ -22,11 +24,11 @@ import static com.google.common.collect.Lists.newArrayList;
*/ */
public class Reply { public class Reply {
public final List<Predicate<Update>> conditions; public final List<Predicate<Update>> conditions;
public final Consumer<Update> action; public final BiConsumer<BaseAbilityBot, Update> action;
private boolean statsEnabled; private boolean statsEnabled;
private String name; private String name;
Reply(List<Predicate<Update>> conditions, Consumer<Update> action) { Reply(List<Predicate<Update>> conditions, BiConsumer<BaseAbilityBot, Update> action) {
this.conditions = ImmutableList.<Predicate<Update>>builder() this.conditions = ImmutableList.<Predicate<Update>>builder()
.addAll(conditions) .addAll(conditions)
.build(); .build();
@ -34,6 +36,29 @@ public class Reply {
statsEnabled = false; statsEnabled = false;
} }
Reply(List<Predicate<Update>> conditions, BiConsumer<BaseAbilityBot, Update> action, String name) {
this(conditions, action);
if (Objects.nonNull(name)) {
enableStats(name);
}
}
/**
* @deprecated Please use {@link #Reply(List, BiConsumer)}
*/
@Deprecated
Reply(List<Predicate<Update>> conditions, Consumer<Update> action) {
this.conditions = ImmutableList.<Predicate<Update>>builder()
.addAll(conditions)
.build();
this.action = ((baseAbilityBot, update) -> action.accept(update));
statsEnabled = false;
}
/**
* @deprecated Please use {@link #Reply(List, BiConsumer, String)}
*/
@Deprecated
Reply(List<Predicate<Update>> conditions, Consumer<Update> action, String name) { Reply(List<Predicate<Update>> conditions, Consumer<Update> action, String name) {
this(conditions, action); this(conditions, action);
if (Objects.nonNull(name)) { if (Objects.nonNull(name)) {
@ -41,10 +66,27 @@ public class Reply {
} }
} }
public static Reply of(BiConsumer<BaseAbilityBot, Update> action, List<Predicate<Update>> conditions) {
return new Reply(conditions, action);
}
@SafeVarargs
public static Reply of(BiConsumer<BaseAbilityBot, Update> action, Predicate<Update>... conditions) {
return Reply.of(action, newArrayList(conditions));
}
/**
* @deprecated Please use {@link #of(BiConsumer, List)}
*/
@Deprecated
public static Reply of(Consumer<Update> action, List<Predicate<Update>> conditions) { public static Reply of(Consumer<Update> action, List<Predicate<Update>> conditions) {
return new Reply(conditions, action); return new Reply(conditions, action);
} }
/**
* @deprecated Please use {@link #of(BiConsumer, Predicate[])}
*/
@Deprecated
@SafeVarargs @SafeVarargs
public static Reply of(Consumer<Update> action, Predicate<Update>... conditions) { public static Reply of(Consumer<Update> action, Predicate<Update>... conditions) {
return Reply.of(action, newArrayList(conditions)); return Reply.of(action, newArrayList(conditions));
@ -56,15 +98,15 @@ public class Reply {
return conditions.stream().reduce(true, stateAnd, Boolean::logicalAnd); return conditions.stream().reduce(true, stateAnd, Boolean::logicalAnd);
} }
public void actOn(Update update) { public void actOn(BaseAbilityBot bot, Update update) {
action.accept(update); action.accept(bot, update);
} }
public List<Predicate<Update>> conditions() { public List<Predicate<Update>> conditions() {
return conditions; return conditions;
} }
public Consumer<Update> action() { public BiConsumer<BaseAbilityBot, Update> action() {
return action; return action;
} }

View File

@ -1,6 +1,7 @@
package org.telegram.abilitybots.api.objects; package org.telegram.abilitybots.api.objects;
import org.jetbrains.annotations.NotNull; import org.jetbrains.annotations.NotNull;
import org.telegram.abilitybots.api.bot.BaseAbilityBot;
import org.telegram.abilitybots.api.db.DBContext; import org.telegram.abilitybots.api.db.DBContext;
import org.telegram.abilitybots.api.util.AbilityUtils; import org.telegram.abilitybots.api.util.AbilityUtils;
import org.telegram.telegrambots.meta.api.objects.Update; import org.telegram.telegrambots.meta.api.objects.Update;
@ -10,6 +11,7 @@ import java.util.HashSet;
import java.util.List; import java.util.List;
import java.util.Set; import java.util.Set;
import java.util.concurrent.atomic.AtomicInteger; import java.util.concurrent.atomic.AtomicInteger;
import java.util.function.BiConsumer;
import java.util.function.Consumer; import java.util.function.Consumer;
import java.util.function.Predicate; import java.util.function.Predicate;
import java.util.stream.Stream; import java.util.stream.Stream;
@ -20,7 +22,7 @@ public class ReplyFlow extends Reply {
private final Set<Reply> nextReplies; private final Set<Reply> nextReplies;
private ReplyFlow(List<Predicate<Update>> conditions, Consumer<Update> action, Set<Reply> nextReplies, String name) { private ReplyFlow(List<Predicate<Update>> conditions, BiConsumer<BaseAbilityBot, Update> action, Set<Reply> nextReplies, String name) {
super(conditions, action, name); super(conditions, action, name);
this.nextReplies = nextReplies; this.nextReplies = nextReplies;
} }
@ -48,7 +50,7 @@ public class ReplyFlow extends Reply {
private final DBContext db; private final DBContext db;
private final int id; private final int id;
private List<Predicate<Update>> conds; private List<Predicate<Update>> conds;
private Consumer<Update> action; private BiConsumer<BaseAbilityBot, Update> action;
private Set<Reply> nextReplies; private Set<Reply> nextReplies;
private String name; private String name;
@ -63,7 +65,16 @@ public class ReplyFlow extends Reply {
this(db, replyCounter.getAndIncrement()); this(db, replyCounter.getAndIncrement());
} }
/**
* @deprecated Please use {@link #action(BiConsumer)}
*/
@Deprecated
public ReplyFlowBuilder action(Consumer<Update> action) { public ReplyFlowBuilder action(Consumer<Update> action) {
this.action = (bot, update) -> action.accept(update);
return this;
}
public ReplyFlowBuilder action(BiConsumer<BaseAbilityBot, Update> action) {
this.action = action; this.action = action;
return this; return this;
} }
@ -80,7 +91,7 @@ public class ReplyFlow extends Reply {
public ReplyFlowBuilder next(Reply nextReply) { public ReplyFlowBuilder next(Reply nextReply) {
List<Predicate<Update>> statefulConditions = toStateful(nextReply.conditions()); List<Predicate<Update>> statefulConditions = toStateful(nextReply.conditions());
Consumer<Update> statefulAction = nextReply.action().andThen(upd -> { BiConsumer<BaseAbilityBot, Update> statefulAction = nextReply.action().andThen((unused, upd) -> {
Long chatId = AbilityUtils.getChatId(upd); Long chatId = AbilityUtils.getChatId(upd);
db.<Long, Integer>getMap(STATES).remove(chatId); db.<Long, Integer>getMap(STATES).remove(chatId);
}); });
@ -100,16 +111,16 @@ public class ReplyFlow extends Reply {
public ReplyFlow build() { public ReplyFlow build() {
if (action == null) if (action == null)
action = upd -> {}; action = (bot, upd) -> {};
Consumer<Update> statefulAction; BiConsumer<BaseAbilityBot, Update> statefulAction;
if (nextReplies.size() > 0) { if (nextReplies.size() > 0) {
statefulAction = action.andThen(upd -> { statefulAction = action.andThen((bot, upd) -> {
Long chatId = AbilityUtils.getChatId(upd); Long chatId = AbilityUtils.getChatId(upd);
db.<Long, Integer>getMap(STATES).put(chatId, id); db.<Long, Integer>getMap(STATES).put(chatId, id);
}); });
} else { } else {
statefulAction = action.andThen(upd -> { statefulAction = action.andThen((bot, upd) -> {
Long chatId = AbilityUtils.getChatId(upd); Long chatId = AbilityUtils.getChatId(upd);
db.<Long, Integer>getMap(STATES).remove(chatId); db.<Long, Integer>getMap(STATES).remove(chatId);
}); });

View File

@ -210,7 +210,7 @@ public class AbilityBotTest {
// Support for null parameter matching since due to mocking API changes // Support for null parameter matching since due to mocking API changes
when(sender.downloadFile(ArgumentMatchers.<File>isNull())).thenReturn(backupFile); when(sender.downloadFile(ArgumentMatchers.<File>isNull())).thenReturn(backupFile);
defaultAbs.recoverDB().replies().get(0).actOn(update); defaultAbs.recoverDB().replies().get(0).actOn(bot, update);
verify(silent, times(1)).send(RECOVER_SUCCESS, GROUP_ID); verify(silent, times(1)).send(RECOVER_SUCCESS, GROUP_ID);
assertEquals(db.getSet(TEST), newHashSet(TEST), "Bot recovered but the DB is still not in sync"); assertEquals(db.getSet(TEST), newHashSet(TEST), "Bot recovered but the DB is still not in sync");

View File

@ -43,8 +43,8 @@ public class DefaultBot extends AbilityBot {
return getDefaultBuilder() return getDefaultBuilder()
.name(DEFAULT) .name(DEFAULT)
.info("dis iz default command") .info("dis iz default command")
.reply(Reply.of(upd -> silent.send("reply", upd.getMessage().getChatId()), MESSAGE, update -> update.getMessage().getText().equals("must reply")).enableStats("mustreply")) .reply(Reply.of((bot, upd) -> silent.send("reply", upd.getMessage().getChatId()), MESSAGE, update -> update.getMessage().getText().equals("must reply")).enableStats("mustreply"))
.reply(upd -> silent.send("reply", upd.getCallbackQuery().getMessage().getChatId()), CALLBACK_QUERY) .reply((bot, upd) -> silent.send("reply", upd.getCallbackQuery().getMessage().getChatId()), CALLBACK_QUERY)
.build(); .build();
} }
@ -75,7 +75,7 @@ public class DefaultBot extends AbilityBot {
public Reply channelPostReply() { public Reply channelPostReply() {
return Reply.of( return Reply.of(
upd -> silent.send("test channel post", upd.getChannelPost().getChatId()), Flag.CHANNEL_POST (bot, upd) -> silent.send("test channel post", upd.getChannelPost().getChatId()), Flag.CHANNEL_POST
); );
} }

View File

@ -163,19 +163,19 @@ public class ReplyFlowTest {
} }
public ReplyFlow directionFlow() { public ReplyFlow directionFlow() {
Reply saidLeft = Reply.of(upd -> silent.send("Sir, I have gone left.", getChatId(upd)), Reply saidLeft = Reply.of((bot, upd) -> silent.send("Sir, I have gone left.", getChatId(upd)),
hasMessageWith("go left or else")); hasMessageWith("go left or else"));
ReplyFlow leftflow = ReplyFlow.builder(db, 2) ReplyFlow leftflow = ReplyFlow.builder(db, 2)
.action(upd -> silent.send("I don't know how to go left.", getChatId(upd))) .action((bot, upd) -> silent.send("I don't know how to go left.", getChatId(upd)))
.onlyIf(hasMessageWith("left")) .onlyIf(hasMessageWith("left"))
.next(saidLeft).build(); .next(saidLeft).build();
Reply saidRight = Reply.of(upd -> silent.send("Sir, I have gone right.", getChatId(upd)), Reply saidRight = Reply.of((bot, upd) -> silent.send("Sir, I have gone right.", getChatId(upd)),
hasMessageWith("right")); hasMessageWith("right"));
return ReplyFlow.builder(db, 1) return ReplyFlow.builder(db, 1)
.action(upd -> silent.send("Command me to go left or right!", getChatId(upd))) .action((bot, upd) -> silent.send("Command me to go left or right!", getChatId(upd)))
.onlyIf(hasMessageWith("wake up")) .onlyIf(hasMessageWith("wake up"))
.next(leftflow) .next(leftflow)
.next(saidRight) .next(saidRight)
@ -184,10 +184,10 @@ public class ReplyFlowTest {
public Reply errantReply() { public Reply errantReply() {
return Reply.of( return Reply.of(
upd -> { (bot, upd) -> {
throw new RuntimeException("Throwing an exception inside the update consumer"); throw new RuntimeException("Throwing an exception inside the update consumer");
}, },
upd -> { (upd) -> {
throw new RuntimeException("Throwing an exception inside the reply conditions (flags)"); throw new RuntimeException("Throwing an exception inside the reply conditions (flags)");
}); });
} }
@ -195,7 +195,7 @@ public class ReplyFlowTest {
public Ability replyFlowsWithAbility() { public Ability replyFlowsWithAbility() {
Reply replyWithVk = ReplyFlow.builder(db, 2) Reply replyWithVk = ReplyFlow.builder(db, 2)
.enableStats("SECOND") .enableStats("SECOND")
.action(upd -> { .action((bot, upd) -> {
silent.send("Second reply", upd.getMessage().getChatId()); silent.send("Second reply", upd.getMessage().getChatId());
}) })
.onlyIf(hasMessageWith("two")) .onlyIf(hasMessageWith("two"))
@ -203,7 +203,7 @@ public class ReplyFlowTest {
Reply replyWithNickname = ReplyFlow.builder(db, 1) Reply replyWithNickname = ReplyFlow.builder(db, 1)
.enableStats("FIRST") .enableStats("FIRST")
.action(upd -> { .action((bot, upd) -> {
silent.send("First reply", upd.getMessage().getChatId()); silent.send("First reply", upd.getMessage().getChatId());
}) })
.onlyIf(hasMessageWith("one")) .onlyIf(hasMessageWith("one"))