【问题标题】:Telegram Bot - Process response from user based on last bot questionTelegram Bot - 根据最后一个机器人问题处理用户的响应
【发布时间】:2020-04-08 00:03:16
【问题描述】:

我希望我的 Telegram 机器人根据机器人提出的最后一个问题来处理用户输入。基本上,流程是这样的:

  • 用户调用 /authenticate 命令
  • 机器人请求电子邮件
  • 用户发送他的电子邮件
  • Bot 回复了一条关于向用户电子邮件发送验证码以进行确认的消息,并要求用户在聊天中输入验证码
  • 用户输入代码
  • Bot 验证用户代码,用户通过身份验证并开始接收通知

问题是:我怎么知道用户正在回答这个流程中的特定机器人问题?

我想到了两种方法:

  • 发送带有强制回复选项的消息,以便用户必须回复机器人问题。这会将用户响应的消息发回给我,以便我可以比较机器人消息字符串以查看答案。

  • 将最后一条 bot 消息存储在某处,然后当消息到达时,检查最后一条 bot 消息是什么,并假设用户消息是响应。

有没有更好的方法?我正在使用带有telegrambots library 的Java。

【问题讨论】:

    标签: java telegram telegram-bot


    【解决方案1】:

    通常要具体说明您向电报机器人发送的内容,您将使用命令与机器人通信(电报命令以 / 开头,并且每个命令在服务器端都有一个特定的命令处理程序),不幸的是,所以到目前为止,没有办法将额外的参数与命令一起发送到电报机器人,您可以使用以下对话中提到的解决方法之一: How do I have my Bot respond with arguments?

    如果您不熟悉使用java的bot命令,请参考以下示例: https://www.programcreek.com/java-api-examples/?api=org.telegram.telegrambots.bots.AbsSender

    我希望这会有所帮助。

    【讨论】:

    • 是的,发送命令时不能使用参数。但我想做的是模仿 BotFather 在创建机器人时所做的流程。
    【解决方案2】:

    开发电报机器人时最流行的任务之一。
    有一个解决方案,您需要为 Laravel 安装 shapoapps/multibot_driver 包(composer 需要 shapoapps/multibot_driver)。
    这个软件包有一个会话管理器,类似于网页用户的会话。在每条用户消息之后,您将用户输入记录到服务器端的会话中。当收到每个新请求(消息)时,您从会话中读取数据并构建逻辑。
    这是文档 - https://github.com/shapoapps/multibot_driver

    【讨论】:

    • 我正在使用 Java :(
    【解决方案3】:

    由于很难找到可以引导我找到解决方案的想法(在 Java 中),为了未来的 Java googlers,我将在这里分享我的想法。我将 telegrambots library 与 Spring Boot/Data 一起使用。

    实现此流程的最佳方式是在数据库中保存状态。为此,使用消息的唯一聊天 ID 将聊天与另一个聊天区分开来

    这是 Java 实现的相关部分(逻辑几乎适用于任何语言):

    持有与系统用户相关的 Telegram 聊天信息的实体。

    @Entity
    @Table(name = "user_bot")
    public class UserBot implements Serializable {
    
        @Id
        @GeneratedValue(strategy = GenerationType.IDENTITY)
        private Integer id;
    
        @Column(name = "chat_id", unique = true, nullable = false, length = 255)
        private String chatId;
    
        @Column(name = "bot_verification_code", length = 6)
        private String botVerificationCode;
    
        @Enumerated
        @Column(name = "last_bot_state", columnDefinition = "SMALLINT DEFAULT NULL")
        private BotState lastBotState;
    
        @Column(columnDefinition = "TINYINT(1)")
        private boolean verified;
    
        @JoinColumn(name = "user_id", referencedColumnName = "id")
        @ManyToOne(fetch = FetchType.EAGER)
        private User user;
    }
    

    代表所有可能的机器人响应(状态)的枚举。

    public enum BotState {
        // Respostas do bot que representam estados
        AUTH_STEP_1("Muito bem. Qual é o seu e-mail no sistema?"), AUTH_STEP_2("Enviei um código para o seu e-mail. Por favor, digite-o aqui."),
        NO_STATE("");
    
        private final String state;
    
        private BotState(String state) {
            this.state = state;
        }
    
        @Override
        public String toString() {
            return this.state;
        }
    }
    

    接收消息并做出相应响应的服务。

    @Service
    public class TelegramBotService extends TelegramLongPollingBot {
    
        @Autowired
        private CodeUtils codeUtils;
    
        @Autowired
        private UserBotRepository userBotRepository;
    
        @Autowired
        private UserRepository userRepository;
    
        @Value("${telegram.bot.username}")
        private String botUsername;
    
        @Value("${telegram.bot.token}")
        private String botToken;
    
        @PostConstruct
        public void registerBot() {
            TelegramBotsApi botsApi = new TelegramBotsApi();
            try {
                botsApi.registerBot(this);
            } catch (TelegramApiException e) {
                e.printStackTrace();
            }
        }
    
        @Override
        public void onUpdateReceived(Update update) {
            if (update.hasMessage() && update.getMessage().hasText()) {
                String receivedMessage = update.getMessage().getText();
                SendMessage sendMessage = null;
    
                // TODO: futuramente, tratar casos onde um usuário chama um comando sem ainda estar autenticado
                switch (receivedMessage) {
                    case "/autenticar":
                        sendMessage = handleAuthentication(update);
                        break;
                    default:
                        // Quando nenhum comando atender, será um texto a ser checado de acordo com o estado anterior
                        sendMessage = checkState(update);
                }
    
                try {
                    execute(sendMessage);
                } catch (TelegramApiException e) {
                    codeUtils.log(e.getMessage(), this);
                }
            }
        }
    
        private SendMessage handleAuthentication(Update update) {
            SendMessage sendMessage = new SendMessage()
                    .setChatId(update.getMessage().getChatId())
                    .setText(BotState.AUTH_STEP_1.toString());
    
            UserBot userBot = userBotRepository.findByChatId(update.getMessage().getChatId().toString());
    
            if (userBot == null) {
                userBot = new UserBot();
                userBot.setChatId(update.getMessage().getChatId().toString());
                userBot.setLastBotState(BotState.AUTH_STEP_1);
            } else if (userBot.isVerified()) {
                // Um texto simples enviado no sendMessage indica o fim de um fluxo
                sendMessage.setText("Este aparelho já está autenticado no sistema.");
                userBot.setLastBotState(null);
            }
    
            userBotRepository.save(userBot);
            return sendMessage;
        }
    
        // Checa o estado anterior do bot em relação ao chatId recebido
        private SendMessage checkState(Update update) {
            UserBot userBot = userBotRepository.findByChatId(update.getMessage().getChatId().toString());
            SendMessage sendMessage = null;
    
            if (userBot == null || userBot.getLastBotState() == null)
                return sendDefaultMessage(update);
    
            switch (Optional.ofNullable(userBot.getLastBotState()).orElse(BotState.NO_STATE)) {
                case AUTH_STEP_1:
                    sendMessage = sendCode(update);
                    break;
                case AUTH_STEP_2:
                    sendMessage = validateCode(update);
                    break;
                default:
                    sendMessage = sendDefaultMessage(update);
            }
    
            return sendMessage;
        }
    
        // Grava o código no banco e envia para o e-mail do usuário
        private SendMessage sendCode(Update update) {
            User user = userRepository.findByEmail(update.getMessage().getText().toLowerCase());
            SendMessage sendMessage = new SendMessage(update.getMessage().getChatId(), "");
    
            if (user == null)
                sendMessage.setText("Não encontrei nenhum usuário no sistema com este e-mail :(");
            else {
                UserBot userBot = userBotRepository.findByChatId(update.getMessage().getChatId().toString());
    
                String verificationCode = Integer.toString(new Random().nextInt(899999) + 100000);
                String text = "Este é um e-mail automático de verificação de identidade. Informe este código para o bot do Telegram: " + verificationCode;
                codeUtils.sendEmail(new String[]{user.getEmail()}, "CCR Laudos - Código de Verificação", text);
    
                // Associa a conversação ao usuário, mas a validade depende da flag verified
                userBot.setUser(user);
                userBot.setBotVerificationCode(verificationCode);
                userBot.setLastBotState(BotState.AUTH_STEP_2);
                userBotRepository.save(userBot);
    
                sendMessage.setText(BotState.AUTH_STEP_2.toString());
            }
    
            return sendMessage;
        }
    
        // Checa se o código informado foi o mesmo passado por e-mail para o usuário a fim de autenticá-lo
        private SendMessage validateCode(Update update) {
            UserBot userBot = userBotRepository.findByChatId(update.getMessage().getChatId().toString());
            SendMessage sendMessage = new SendMessage(update.getMessage().getChatId(), "");
    
            if (update.getMessage().getText().equals(userBot.getBotVerificationCode())) {
                userBot.setVerified(true);
                sendMessage.setText("O aparelho foi autenticado com sucesso. Você passará a receber notificações do sistema.");
            } else {
                userBot.setUser(null);
                sendMessage.setText("Código inválido.");
            }
    
            userBotRepository.save(userBot);
            return sendMessage;
        }
    
        private SendMessage sendDefaultMessage(Update update) {
            String markdownMessage = "Não entendi \ud83e\udd14 \n"
                    + "Que tal tentar um comando digitando */* ?";
            return new SendMessage(update.getMessage().getChatId(), markdownMessage).setParseMode(ParseMode.MARKDOWN);
        }
    
        @Override
        public String getBotUsername() {
            return this.botUsername;
        }
    
        @Override
        public String getBotToken() {
            return this.botToken;
        }
    }
    

    实现的流程是:

    1. 用户发送 /authenticate。

    2. 系统对设备一无所知,因此存储聊天 id 和最后状态。 最后一个状态将是对用户的响应。系统要求用户的电子邮件。

    3. 用户发送他的电子邮件。

    4. 文本未被识别为命令,因此系统检查是否存在与此聊天 ID 相关的最后状态。如果存在先前的状态,则使用传入文本作为此状态方法的参数。系统向用户的电子邮件发送一个代码并要求它。

    5. 用户发送代码。

    6. 如果代码正确,系统会再次检查之前的状态并对用户进行身份验证。

    就是这样!希望它可以帮助某人。

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 2015-11-16
      • 2022-10-18
      • 2018-08-11
      • 2016-11-03
      • 2018-10-05
      • 1970-01-01
      • 2020-11-11
      • 2018-10-15
      相关资源
      最近更新 更多