mirror of
				https://github.com/zulip/zulip.git
				synced 2025-11-03 21:43:21 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			122 lines
		
	
	
		
			4.5 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			122 lines
		
	
	
		
			4.5 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
import json
 | 
						|
import logging
 | 
						|
import os
 | 
						|
import signal
 | 
						|
import sys
 | 
						|
import time
 | 
						|
import re
 | 
						|
import importlib
 | 
						|
from zerver.lib.actions import internal_send_private_message, \
 | 
						|
    internal_send_stream_message, internal_send_huddle_message
 | 
						|
from zerver.models import UserProfile, get_user
 | 
						|
from zerver.lib.bot_storage import get_bot_storage, set_bot_storage, \
 | 
						|
    is_key_in_bot_storage, get_bot_storage_size, remove_bot_storage
 | 
						|
from zerver.lib.bot_config import get_bot_config, ConfigError
 | 
						|
from zerver.lib.integrations import EMBEDDED_BOTS
 | 
						|
 | 
						|
import configparser
 | 
						|
 | 
						|
if False:
 | 
						|
    from mypy_extensions import NoReturn
 | 
						|
from typing import Any, Optional, List, Dict
 | 
						|
from types import ModuleType
 | 
						|
 | 
						|
our_dir = os.path.dirname(os.path.abspath(__file__))
 | 
						|
 | 
						|
from zulip_bots.lib import RateLimit
 | 
						|
 | 
						|
def get_bot_handler(service_name: str) -> Any:
 | 
						|
 | 
						|
    # Check that this service is present in EMBEDDED_BOTS, add exception handling.
 | 
						|
    is_present_in_registry = any(service_name == embedded_bot_service.name for
 | 
						|
                                 embedded_bot_service in EMBEDDED_BOTS)
 | 
						|
    if not is_present_in_registry:
 | 
						|
        return None
 | 
						|
    bot_module_name = 'zulip_bots.bots.%s.%s' % (service_name, service_name)
 | 
						|
    bot_module = importlib.import_module(bot_module_name)  # type: Any
 | 
						|
    return bot_module.handler_class()
 | 
						|
 | 
						|
 | 
						|
class StateHandler:
 | 
						|
    storage_size_limit = 10000000   # type: int # TODO: Store this in the server configuration model.
 | 
						|
 | 
						|
    def __init__(self, user_profile: UserProfile) -> None:
 | 
						|
        self.user_profile = user_profile
 | 
						|
        self.marshal = lambda obj: json.dumps(obj)
 | 
						|
        self.demarshal = lambda obj: json.loads(obj)
 | 
						|
 | 
						|
    def get(self, key: str) -> str:
 | 
						|
        return self.demarshal(get_bot_storage(self.user_profile, key))
 | 
						|
 | 
						|
    def put(self, key: str, value: str) -> None:
 | 
						|
        set_bot_storage(self.user_profile, [(key, self.marshal(value))])
 | 
						|
 | 
						|
    def remove(self, key: str) -> None:
 | 
						|
        remove_bot_storage(self.user_profile, [key])
 | 
						|
 | 
						|
    def contains(self, key: str) -> bool:
 | 
						|
        return is_key_in_bot_storage(self.user_profile, key)
 | 
						|
 | 
						|
class EmbeddedBotQuitException(Exception):
 | 
						|
    pass
 | 
						|
 | 
						|
class EmbeddedBotHandler:
 | 
						|
    def __init__(self, user_profile: UserProfile) -> None:
 | 
						|
        # Only expose a subset of our UserProfile's functionality
 | 
						|
        self.user_profile = user_profile
 | 
						|
        self._rate_limit = RateLimit(20, 5)
 | 
						|
        self.full_name = user_profile.full_name
 | 
						|
        self.email = user_profile.email
 | 
						|
        self.storage = StateHandler(user_profile)
 | 
						|
 | 
						|
    def send_message(self, message: Dict[str, Any]) -> None:
 | 
						|
        if not self._rate_limit.is_legal():
 | 
						|
            self._rate_limit.show_error_and_exit()
 | 
						|
 | 
						|
        if message['type'] == 'stream':
 | 
						|
            internal_send_stream_message(self.user_profile.realm, self.user_profile, message['to'],
 | 
						|
                                         message['subject'], message['content'])
 | 
						|
            return
 | 
						|
 | 
						|
        assert message['type'] == 'private'
 | 
						|
        # Ensure that it's a comma-separated list, even though the
 | 
						|
        # usual 'to' field could be either a List[str] or a str.
 | 
						|
        recipients = ','.join(message['to']).split(',')
 | 
						|
 | 
						|
        if len(message['to']) == 1:
 | 
						|
            recipient_user = get_user(recipients[0], self.user_profile.realm)
 | 
						|
            internal_send_private_message(self.user_profile.realm, self.user_profile,
 | 
						|
                                          recipient_user, message['content'])
 | 
						|
        else:
 | 
						|
            internal_send_huddle_message(self.user_profile.realm, self.user_profile,
 | 
						|
                                         recipients, message['content'])
 | 
						|
 | 
						|
    def send_reply(self, message: Dict[str, Any], response: str) -> None:
 | 
						|
        if message['type'] == 'private':
 | 
						|
            self.send_message(dict(
 | 
						|
                type='private',
 | 
						|
                to=[x['email'] for x in message['display_recipient']],
 | 
						|
                content=response,
 | 
						|
                sender_email=message['sender_email'],
 | 
						|
            ))
 | 
						|
        else:
 | 
						|
            self.send_message(dict(
 | 
						|
                type='stream',
 | 
						|
                to=message['display_recipient'],
 | 
						|
                subject=message['subject'],
 | 
						|
                content=response,
 | 
						|
                sender_email=message['sender_email'],
 | 
						|
            ))
 | 
						|
 | 
						|
    # The bot_name argument exists only to comply with ExternalBotHandler.get_config_info().
 | 
						|
    def get_config_info(self, bot_name: str, optional: bool=False) -> Dict[str, str]:
 | 
						|
        try:
 | 
						|
            return get_bot_config(self.user_profile)
 | 
						|
        except ConfigError:
 | 
						|
            if optional:
 | 
						|
                return dict()
 | 
						|
            raise
 | 
						|
 | 
						|
    def quit(self, message: str= "") -> None:
 | 
						|
        raise EmbeddedBotQuitException(message)
 |