如何限制对电报机器人的访问

Posted

技术标签:

【中文标题】如何限制对电报机器人的访问【英文标题】:How To Limit Access To A Telegram Bot 【发布时间】:2016-05-23 22:54:09 【问题描述】:

当我向我的 Telegram Bot 发送消息时,它会毫无问题地响应。

我想限制访问权限,只有我自己可以向它发送消息。

我该怎么做?

【问题讨论】:

【参考方案1】:

按字段过滤消息update.message.from.id

【讨论】:

【参考方案2】:

与您的机器人开始对话,并向其发送消息。这将为包含您对话的消息和聊天 ID 的机器人排队更新。

要查看最近的更新,请调用 getUpdates 方法。这是通过向 URL https://api.telegram.org/bot$TOKEN/getUpdates 发出 HTTP GET 请求来完成的,其中 $TOKEN 是 BotFather 提供的令牌。比如:

"chat":
        "id":12345,
        "first_name":"Bob",
        "last_name":"Jones",
        "username":"bjones",
        "type":"private",
      "date":1452933785,
      "text":"Hi there, bot!"]

一旦您确定了您的聊天 ID,您就可以在您的机器人中编写一段代码,例如:

id_a = [111111,2222222,3333333,4444444,5555555]

    def handle(msg):
        chat_id = msg['chat']['id']
        command = msg['text']
        sender = msg['from']['id']
     if sender in id_a:
    [...]
     else:
           bot.sendMessage(chat_id, 'Forbidden access!')
           bot.sendMessage(chat_id, sender)

【讨论】:

【参考方案3】:

由于此问题与python-telegram-bot有关,因此以下信息与之相关:

当您向机器人的调度程序添加处理程序时,您可以指定各种预构建的过滤器(在docs、github 阅读更多信息),或者您可以创建自定义过滤器以过滤传入的更新。

为了限制特定用户的访问,您需要在初始化处理程序时添加Filters.user(username="@telegramusername"),例如:

dispatcher.add_handler(CommandHandler("start", text_callback, Filters.user(username="@username")))

此处理程序将仅接受来自用户名为 @username 的用户的 /start 命令。

您也可以指定用户 ID 而不是用户名,我强烈建议您这样做,因为后者是非常量的,并且可以随着时间的推移而更改。

【讨论】:

【参考方案4】:

update.message.chat_id 过滤对我有用。 要找到您的聊天 ID,请向您的机器人发送消息并浏览到

https://api.telegram.org/bot$TOKEN/getUpdates

其中$TOKEN 是 BotFather 提供的机器人令牌,如 fdicarlo 的回答中所述,您可以在其中找到 json 结构中的聊天 ID。

【讨论】:

【参考方案5】:

基于python-telegram-bot 代码sn-ps,可以围绕处理程序构建一个简单的包装器:

def restricted(func):
    """Restrict usage of func to allowed users only and replies if necessary"""
    @wraps(func)
    def wrapped(bot, update, *args, **kwargs):
        user_id = update.effective_user.id
        if user_id not in conf['restricted_ids']:
            print("WARNING: Unauthorized access denied for .".format(user_id))
            update.message.reply_text('User disallowed.')
            return  # quit function
        return func(bot, update, *args, **kwargs)
    return wrapped

conf['restricted_ids'] 可能是一个 id 列表,例如[11111111, 22222222].

所以用法如下所示:

@restricted
def bot_start(bot, update):
    """Send a message when the command /start is issued"""
    update.message.reply_text('Hi! This is  speaking.'.format(bot.username))

更新 |对于函数和方法(DRY、python-telegram-bot version 12 and above)

上面的解决方案不能用于类内部的方法(位置参数改变)。有一段时间,我刚刚为方法创建了一个额外的装饰器。然而,人们总是需要记住何时使用哪种装饰器。

在this solution 之上构建,可以构建以下装饰器类:

class restricted(object):
    """
    Decorator class used to restrict usage of commands.
    Sends a "disallowed" reply if necessary. Works on functions and methods.
    """
    def __init__(self, func):
        self._func = func
        self._obj = None
        self._wrapped = None

    def __call__(self, *args, **kwargs):
        if not self._wrapped:
            if self._obj:
                self._wrapped = self._wrap_method(self._func)
                self._wrapped = partial(self._wrapped, self._obj)
            else:
                self._wrapped = self._wrap_function(self._func)
        return self._wrapped(*args, **kwargs)

    def __get__(self, obj, type_=None):
        self._obj = obj
        return self

    def _wrap_method(self, method): # Wrapper called in case of a method
        @wraps(method)
        def inner(self, *args, **kwargs): # `self` is the *inner* class' `self` here
            user_id = args[0].effective_user.id # args[0]: update
            if user_id not in cfg.RESTRICTED_IDS:
                print(f'Unauthorized access denied on method.__name__ ' \
                    f'for user_id : args[0].message.chat.username.')
                args[0].message.reply_text('User disallowed.')
                return None # quit handling command
            return method(self, *args, **kwargs)
        return inner

    def _wrap_function(self, function): # Wrapper called in case of a function
        @wraps(function)
        def inner(*args, **kwargs): # `self` would be the *restricted* class' `self` here
            user_id = args[0].effective_user.id # args[0]: update
            if user_id not in cfg.RESTRICTED_IDS:
                print(f'Unauthorized access denied on function.__name__ ' \
                    f'for user_id : args[0].message.chat.username.')
                args[0].message.reply_text('User disallowed.')
                return None # quit handling command
            return function(*args, **kwargs)
        return inner

然后,正如预期的那样,这适用于函数和方法。它并不完全是 DRY(参见 cmets),但至少是独立的。

【讨论】:

以上是关于如何限制对电报机器人的访问的主要内容,如果未能解决你的问题,请参考以下文章

如何使私人电报机器人只能由其所有者访问?

如何在电报机器人中获得身份验证?

我无法访问我在电报机器人中使用 getupdates 发送的消息

电报机器人可以访问有关用户的哪些信息?

如何使用 python 将电报机器人添加到不同的组?

电报机器人可以读取频道消息吗