commit
0b2ed0e002
13 changed files with 566 additions and 0 deletions
@ -0,0 +1,37 @@ |
|||||||
|
#!/usr/bin/env python |
||||||
|
|
||||||
|
import argparse |
||||||
|
import asyncio |
||||||
|
import logging |
||||||
|
|
||||||
|
from tools.base import CoboBot |
||||||
|
from tools.processors import MessageDispatcher, CommandsDispatcherProcessor, ConnectionDispatcher |
||||||
|
from tools.processors.messages import RandomTrigger, PikaAttackCommand, \ |
||||||
|
RandomCommandTrigger |
||||||
|
|
||||||
|
# setting up argument parser |
||||||
|
parser = argparse.ArgumentParser(description='Le lou bot') |
||||||
|
parser.add_argument('--cookie', type=str, help='usercookie to use') |
||||||
|
parser.add_argument('--channel', type=str, help='channel to watch', default="") |
||||||
|
parser.add_argument('--domain', type=str, help='domain to connect to', default="loult.family") |
||||||
|
parser.add_argument('--port', type=int, help='port on which to connect the socket', default=80) |
||||||
|
parser.add_argument('--method', type=str, help='http or https', default="https") |
||||||
|
|
||||||
|
|
||||||
|
# setting up the various dispatchers |
||||||
|
coco_commands = CommandsDispatcherProcessor([], "coco", default_response="de?") |
||||||
|
|
||||||
|
root_messages_dispatcher = MessageDispatcher([coco_commands]) |
||||||
|
|
||||||
|
connections_dispatcher = ConnectionDispatcher([]) |
||||||
|
|
||||||
|
if __name__ == "__main__": |
||||||
|
logging.getLogger().setLevel(logging.INFO) |
||||||
|
args = parser.parse_args() |
||||||
|
|
||||||
|
cocobot = CoboBot(args.cookie, args.channel, args.domain, args.port, args.method, |
||||||
|
root_messages_dispatcher, connections_dispatcher) |
||||||
|
asyncio.get_event_loop().run_until_complete(pikabot.listen()) |
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@ -0,0 +1,97 @@ |
|||||||
|
import html |
||||||
|
import json |
||||||
|
import logging |
||||||
|
|
||||||
|
import websockets |
||||||
|
|
||||||
|
from tools.commons import AbstractResponse, Message, BotMessage, AttackCommand, Sound, UserList |
||||||
|
from tools.processors import MessageDispatcher, ConnectionDispatcher |
||||||
|
|
||||||
|
|
||||||
|
class CoboBot: |
||||||
|
|
||||||
|
def __init__(self, cookie : str, channel : str, domain : str, port : int, method : str, |
||||||
|
messages_dispatcher : MessageDispatcher, |
||||||
|
connect_dispatcher : ConnectionDispatcher): |
||||||
|
|
||||||
|
# setting up variables required by the server. The default is a Kabutops on the main lou server, I think |
||||||
|
self.cookie = cookie |
||||||
|
self.channel = "" if channel == "root" else channel |
||||||
|
self.domain = domain |
||||||
|
self.port = port |
||||||
|
self.method = method |
||||||
|
self.msg_dispatch = messages_dispatcher |
||||||
|
self.cnt_dispatch = connect_dispatcher |
||||||
|
self.user_list = None # type: UserList |
||||||
|
|
||||||
|
async def _send_message(self, message): |
||||||
|
if isinstance(message, dict): |
||||||
|
await self.socket.send(json.dumps(message)) |
||||||
|
elif isinstance(message, bytes): |
||||||
|
await self.socket.send(message) |
||||||
|
|
||||||
|
async def _dispatch_response(self, response_obj : AbstractResponse): |
||||||
|
if isinstance(response_obj, (Message, BotMessage, AttackCommand)): |
||||||
|
await self._send_message(response_obj.to_dict()) |
||||||
|
elif isinstance(response_obj, Sound): |
||||||
|
await self._send_message(response_obj.get_bytes()) |
||||||
|
|
||||||
|
async def _on_connect(self, msg_data): |
||||||
|
# registering the user to the user list |
||||||
|
self.user_list.add_user(msg_data["userid"], msg_data["params"]) |
||||||
|
logging.info("%s connected" % self.user_list.name(msg_data["userid"])) |
||||||
|
message = self.cnt_dispatch.dispatch(msg_data["userid"], self.user_list) |
||||||
|
await self._send_message(message) |
||||||
|
|
||||||
|
async def _on_disconnect(self, msg_data): |
||||||
|
# removing the user from the userlist |
||||||
|
logging.info("%s disconnected" % self.user_list.name(msg_data["userid"])) |
||||||
|
self.user_list.del_user(msg_data["userid"]) |
||||||
|
|
||||||
|
async def _on_message(self, msg_data): |
||||||
|
msg_data["msg"] = html.unescape(msg_data["msg"]) # removing HTML shitty encoding |
||||||
|
# logging the message to the DB |
||||||
|
logging.info("%s says : \"%s\"" % (self.user_list.name(msg_data["userid"]), msg_data["msg"])) |
||||||
|
|
||||||
|
response = None |
||||||
|
# dispatching the message to the processors. If there's a response, send it to the chat |
||||||
|
if not self.user_list.itsme(msg_data["userid"]): |
||||||
|
response = self.msg_dispatch.dispatch(msg_data["msg"], msg_data["userid"], self.user_list) |
||||||
|
|
||||||
|
if isinstance(response, list): |
||||||
|
for response_obj in response: |
||||||
|
await self._dispatch_response(response_obj) |
||||||
|
elif isinstance(response, AbstractResponse): |
||||||
|
await self._dispatch_response(response) |
||||||
|
|
||||||
|
async def listen(self): |
||||||
|
if self.method == "https": |
||||||
|
socket_address = 'wss://%s/socket/%s' % (self.domain, self.channel) |
||||||
|
else: |
||||||
|
socket_address = 'ws://%s:%i/socket/%s' % (self.domain, self.port, self.channel) |
||||||
|
logging.info("Listening to socket on %s" % socket_address) |
||||||
|
async with websockets.connect(socket_address, |
||||||
|
extra_headers={"cookie": "id=%s" % self.cookie}) as websocket: |
||||||
|
self.socket = websocket |
||||||
|
while True: |
||||||
|
msg = await websocket.recv() |
||||||
|
websocket.recv() |
||||||
|
if type(msg) != bytes: |
||||||
|
msg_data = json.loads(msg, encoding="utf-8") |
||||||
|
msg_type = msg_data.get("type", "") |
||||||
|
if msg_type == "userlist": |
||||||
|
self.user_list = UserList(msg_data["users"]) |
||||||
|
logging.info(str(self.user_list)) |
||||||
|
|
||||||
|
elif msg_type == "msg": |
||||||
|
await self._on_message(msg_data) |
||||||
|
|
||||||
|
elif msg_type == "connect": |
||||||
|
await self._on_connect(msg_data) |
||||||
|
|
||||||
|
elif msg_type == "disconnect": |
||||||
|
await self._on_disconnect(msg_data) |
||||||
|
|
||||||
|
else: |
||||||
|
logging.debug("Received sound file") |
||||||
|
|
||||||
@ -0,0 +1,58 @@ |
|||||||
|
from typing import List |
||||||
|
from .requests import LoginRequest, PostLoginRequest |
||||||
|
|
||||||
|
import logging |
||||||
|
|
||||||
|
|
||||||
|
class Interlocutor: |
||||||
|
|
||||||
|
def __init__(self, nick: str, age:int, city: str, is_male: bool, conv_id: str): |
||||||
|
self.nick = nick |
||||||
|
self.age = age |
||||||
|
self.is_male = is_male |
||||||
|
self.city = city |
||||||
|
self.id = conv_id |
||||||
|
|
||||||
|
@classmethod |
||||||
|
def from_string(cls, str): |
||||||
|
# 47130922100412004Rastapopoulos |
||||||
|
# 47 (age) 1 (sexe) 30922 (city id) 100412(conv id) |
||||||
|
age = str[:2] |
||||||
|
is_male = int(str[2:3]) in (1, 6) |
||||||
|
city_id = str[3:8] |
||||||
|
conv_id = str[8:14] |
||||||
|
nick = str[17:] |
||||||
|
return cls(nick, age, city_id, is_male, conv_id) |
||||||
|
|
||||||
|
def __eq__(self, other): |
||||||
|
return other.nick == self.nick |
||||||
|
|
||||||
|
|
||||||
|
class CocoClient: |
||||||
|
|
||||||
|
def __init__(self, nick: str, age: int, is_female: bool, zip_code: str): |
||||||
|
self.nick = nick |
||||||
|
self.age = age |
||||||
|
self.is_female = is_female |
||||||
|
self.zip_code = zip_code |
||||||
|
self.interlocutors = [] # type: List[Interlocutor] |
||||||
|
|
||||||
|
self.user_id = None # type:str |
||||||
|
self.user_pass = None # type:str |
||||||
|
|
||||||
|
def connect(self): |
||||||
|
login_req = LoginRequest(self.nick, self.age, self.is_female, self.zip_code) |
||||||
|
self.user_id, self.user_pass = login_req.retrieve() |
||||||
|
logging.info("Logged in to coco as %s" % self.nick) |
||||||
|
post_login_req = PostLoginRequest(self.user_id, self.user_pass) |
||||||
|
post_login_req.retrieve() |
||||||
|
logging.info("Post login successful") |
||||||
|
|
||||||
|
def pulse(self): |
||||||
|
pass |
||||||
|
|
||||||
|
def send_msg(self): |
||||||
|
pass |
||||||
|
|
||||||
|
def switch_conv(self, nick: str=None): |
||||||
|
pass |
||||||
@ -0,0 +1,99 @@ |
|||||||
|
from urllib.request import Request, urlopen |
||||||
|
from random import randint, choice, random |
||||||
|
from string import ascii_uppercase |
||||||
|
|
||||||
|
from .tools import get_city_id, coco_cipher, encode_msg |
||||||
|
|
||||||
|
class BaseCocoRequest: |
||||||
|
host = 'http://cloud.coco.fr/' |
||||||
|
headers = {'User-Agent': 'Mozilla/5.0 (X11; Ubuntu; Linux x86_64; rv:59.0) Gecko/20100101 Firefox/59.0', |
||||||
|
'Cookie': '_ga=GA1.2.795717920.1518381607', |
||||||
|
'Host': 'cloud.coco.fr', |
||||||
|
'Referer': 'http://www.coco.fr/chat/index.html'} |
||||||
|
|
||||||
|
def _get_url(self): |
||||||
|
pass |
||||||
|
|
||||||
|
def _parse_response(self, response): |
||||||
|
pass |
||||||
|
|
||||||
|
def retrieve(self): |
||||||
|
req = Request(self._get_url(), headers=self.headers) |
||||||
|
response = urlopen(req).read() |
||||||
|
cleaned = response.decode("utf-8")[len("process1('#"):-len("');")] |
||||||
|
return self._parse_response(cleaned) |
||||||
|
|
||||||
|
|
||||||
|
class LoginRequest(BaseCocoRequest): |
||||||
|
|
||||||
|
def __init__(self, nick: str, age: int, is_female: bool, zip_code: str): |
||||||
|
self.nick = nick |
||||||
|
self.age = str(age) |
||||||
|
self.sex = str('2' if is_female else '1') |
||||||
|
self.city = get_city_id(zip_code) |
||||||
|
|
||||||
|
def _get_url(self): |
||||||
|
identifier = str(randint(100000000, 990000000)) + '0' + ''.join(choice(ascii_uppercase) for i in range(20)) |
||||||
|
return self.host + '40' + self.nick + '*' + self.age + self.sex + self.city + identifier |
||||||
|
|
||||||
|
def _parse_response(self, response): |
||||||
|
credentials = response[2:14] |
||||||
|
return credentials[:6], credentials[6:] # user_id and password |
||||||
|
|
||||||
|
|
||||||
|
class LoggedInRequest(BaseCocoRequest): |
||||||
|
|
||||||
|
def __init__(self, user_id, password): |
||||||
|
self.user_id = user_id |
||||||
|
self.password = password |
||||||
|
|
||||||
|
@property |
||||||
|
def token(self): |
||||||
|
return self.user_id + self.password |
||||||
|
|
||||||
|
|
||||||
|
class PostLoginRequest(LoggedInRequest): |
||||||
|
|
||||||
|
client_id_str = "3551366741*0*1aopiig*-940693579*192.168.0.14*0" |
||||||
|
|
||||||
|
def _get_url(self): |
||||||
|
return self.host + '52' + self.token + coco_cipher(self.client_id_str, self.password) |
||||||
|
|
||||||
|
def _parse_response(self, response): |
||||||
|
"""Checks if the post-login was successful""" |
||||||
|
# TODO |
||||||
|
pass |
||||||
|
|
||||||
|
|
||||||
|
class PulseRequest(LoggedInRequest): |
||||||
|
# typical response : |
||||||
|
# process1('#669276787#30130916276787003HotelDiscret#salut_toi#47130922100412004Rastapopoulos#Mes_hommages,_Mademoiselle...#47130922100412004Rastapopoulos#Jamais_un_mari_ne_sera_si_bien_veng*r_que_par_l*8amant_de_sa_femme.#40636427396758003leo913#cam.!7w2738702leo913#396758#'); |
||||||
|
# process1('#669276787#30130916276787003HotelDiscret#chaude=#292223#32130926292223003HDirect#Salut,_te_faire_payer_pour_un_plan_sexe_ca_te_plairais_=#'); |
||||||
|
# process1('#66945630927183748003WolfiSoDentelle#en_manque_de_sommeil_peut_etre_=_#'); |
||||||
|
# ^ le type a 45 balais donc il doit falloir couper après 669 |
||||||
|
# process1('#66929630926396791003Clouds#bonsoir,_comment_vas-tu_=_que_cherches_tu_=#'); |
||||||
|
# idem avec lui, il a 29 ans |
||||||
|
def _get_url(self): |
||||||
|
return self.host + "95" + self.token + "?" + str(random()) |
||||||
|
|
||||||
|
def _parse_response(self, response): |
||||||
|
"""Can either be a single message or several messages""" |
||||||
|
# TODO |
||||||
|
pass |
||||||
|
|
||||||
|
|
||||||
|
class SendMsgRequest(LoggedInRequest): |
||||||
|
|
||||||
|
def __init__(self, user_id, password, conv_id: str, msg: str): |
||||||
|
super().__init__(user_id, password) |
||||||
|
self.conv_id = conv_id |
||||||
|
self.msg = msg |
||||||
|
|
||||||
|
def _get_url(self): |
||||||
|
return self.host + "99" + self.token + self.conv_id + encode_msg(self.msg) |
||||||
|
|
||||||
|
def _parse_response(self, response): |
||||||
|
"""Response to a send message request can either just be #97x or an |
||||||
|
actual message (like in pulse request)""" |
||||||
|
# TODO |
||||||
|
pass |
||||||
@ -0,0 +1,88 @@ |
|||||||
|
from urllib.request import urlopen |
||||||
|
import re |
||||||
|
|
||||||
|
doc = {0: 65, 1: 66, 2: 67, 3: 68, 4: 69, 5: 70, 6: 71, 7: 72, 8: 73, 9: 74, 10: 75, 11: 76, 12: 77, 13: 78, 14: 79, |
||||||
|
15: 80, 16: 81, 17: 82, 18: 83, 19: 84, 20: 101, 21: 102, 22: 103, 23: 104, 24: 105, 25: 106, 26: 107, |
||||||
|
27: 108, 28: 109, 29: 110, 30: 85, 31: 86, 32: 87, 33: 88, 34: 89, 35: 90, 36: 97, 37: 98, 38: 99, 39: 100, |
||||||
|
40: 111, 41: 112, 42: 113, 43: 114, 44: 115, 45: 116, 46: 117, 47: 118, 48: 119, 49: 120, 50: 121, 51: 122, |
||||||
|
52: 48, 53: 49, 54: 50, 55: 51, 56: 52, 57: 53, 58: 54, 59: 55, 60: 56, 61: 57, 62: 43, 63: 47, 64: 61} |
||||||
|
|
||||||
|
|
||||||
|
def coco_cipher(str: str, key: str): |
||||||
|
"""Implementation of coco's weird 'enxo' cipher. key has to be the user's password, |
||||||
|
retrieved from a previous request""" |
||||||
|
|
||||||
|
def none_int(var): |
||||||
|
return var if var is not None else 0 |
||||||
|
|
||||||
|
def safe_get_charcode(s: str, idx: int): |
||||||
|
try: |
||||||
|
return ord(s[idx]) |
||||||
|
except IndexError: |
||||||
|
return None |
||||||
|
|
||||||
|
output, chr1, chr2, chr3 = "", 0, 0, 0 |
||||||
|
enc, revo = {}, {} |
||||||
|
for j in range(65): |
||||||
|
revo[doc[j]] = j |
||||||
|
result = "" |
||||||
|
for i, char_n in enumerate(str): |
||||||
|
result += chr(ord(key[i % len(key)]) ^ ord(char_n)) |
||||||
|
i = 0 |
||||||
|
|
||||||
|
while i < len(str): |
||||||
|
chr1 = safe_get_charcode(result, i) |
||||||
|
i += 1 |
||||||
|
chr2 = safe_get_charcode(result, i) |
||||||
|
i += 1 |
||||||
|
chr3 = safe_get_charcode(result, i) |
||||||
|
i += 1 |
||||||
|
|
||||||
|
enc[0] = none_int(chr1) >> 2 |
||||||
|
enc[1] = ((none_int(chr1) & 3) << 4) | (none_int(chr2) >> 4) |
||||||
|
enc[2] = ((none_int(chr2) & 15) << 2) | (none_int(chr3) >> 6) |
||||||
|
enc[3] = none_int(chr3) & 63 |
||||||
|
if chr2 is None: |
||||||
|
enc[2] = 64 |
||||||
|
enc[3] = 64 |
||||||
|
elif chr3 is None: |
||||||
|
enc[3] = 64 |
||||||
|
|
||||||
|
for j in range(4): |
||||||
|
output += chr(doc[enc[j]]) |
||||||
|
return output |
||||||
|
|
||||||
|
|
||||||
|
def get_city_id(postal_code: str): |
||||||
|
response = str(urlopen("http://www.coco.fr/cocoland/%s.js" % postal_code).read(), 'utf-8') |
||||||
|
first_city_code = re.search(r'[0-9]+', response) |
||||||
|
if first_city_code: |
||||||
|
return first_city_code.group() |
||||||
|
elif 'ERROR' in response: |
||||||
|
raise ValueError('Invalid postal code') |
||||||
|
else: |
||||||
|
RuntimeError('Unexpected output') |
||||||
|
|
||||||
|
|
||||||
|
smilies = [":)", ":(", ";)", ":d", ":-o", ":s", ":$", "*-)", "-)", "^o)", ":p", "(l)", "(v)", ":'(", "(h)", "(f)", |
||||||
|
":@", "(y)", "(n)", "(k)", "gr$", "(a)", "(6)", "(yn)", "+o(", "na$", "oh$", "tr$", "(e)", "sh$", "fu$", |
||||||
|
"nw$", "ba$", "ao$", "db$", "si$", "oo$", "co$", "bi$", "cc$", "ye$", "mo$", "aa$", "ci$", "uu$", "ff$", |
||||||
|
"zz$", "gt$", "ah$", "mm$", "?$", "xx$"] |
||||||
|
|
||||||
|
special_chars = {" ": "~", "!": "!", "\"": "*8", "$": "*7", "%": "*g", "'": "*8", "(": "(", ")": ")", "*": "*s", "=": "*h", |
||||||
|
"?": "=", "@": "*m", "^": "*l", "_": "*0", "€": "*d", "à": "*a", "â": "*k", "ç": "*c", "è": "*e", |
||||||
|
"é": "*r", "ê": "*b", "î": "*i", "ï": "*j", "ô": "*o", "ù": "*f", "û": "*u"} |
||||||
|
|
||||||
|
|
||||||
|
def encode_msg(msg : str): |
||||||
|
"""Encoding the message to coco's weird url-compatible format""" |
||||||
|
for i in range(len(smilies)): |
||||||
|
msg = msg.replace(smilies[i], ';' + str(i).zfill(2)) |
||||||
|
|
||||||
|
for char, replacement in special_chars.items(): |
||||||
|
msg = msg.replace(char, replacement) |
||||||
|
|
||||||
|
return msg |
||||||
|
|
||||||
|
def decode_msg(msg: str): |
||||||
|
pass |
||||||
@ -0,0 +1,84 @@ |
|||||||
|
|
||||||
|
|
||||||
|
class UserList: |
||||||
|
"""Wrapper around the 'currently connected users' dictionary""" |
||||||
|
|
||||||
|
def __init__(self, userlist_data): |
||||||
|
self.users = {user_data["userid"]: user_data for user_data in userlist_data} |
||||||
|
for id, user in self.users.items(): |
||||||
|
if "you" in user["params"] and user["params"]["you"]: |
||||||
|
self.my_id = id |
||||||
|
|
||||||
|
def del_user(self, user_id): |
||||||
|
del self.users[user_id] |
||||||
|
|
||||||
|
def add_user(self, user_id, params): |
||||||
|
self.users[user_id] = {"params": params} |
||||||
|
|
||||||
|
def __getitem__(self, item : str): |
||||||
|
return self.users[item]["params"] |
||||||
|
|
||||||
|
def name(self, user_id): |
||||||
|
return self.users[user_id]["params"]["name"] |
||||||
|
|
||||||
|
def get_all_names(self): |
||||||
|
return [user["params"]["name"] for user in self.users.values()] |
||||||
|
|
||||||
|
def itsme(self, user_id): |
||||||
|
return self.my_id == user_id |
||||||
|
|
||||||
|
@property |
||||||
|
def my_name(self): |
||||||
|
return self.name(self.my_id) |
||||||
|
|
||||||
|
def __str__(self): |
||||||
|
return "Connected users :\n" \ |
||||||
|
"%s" % "\n".join(["\t - %s" % self.name(user_id) for user_id in self.users]) |
||||||
|
|
||||||
|
|
||||||
|
class AbstractResponse: |
||||||
|
pass |
||||||
|
|
||||||
|
|
||||||
|
class Message(AbstractResponse): |
||||||
|
|
||||||
|
def __init__(self, msg: str): |
||||||
|
self.msg = msg |
||||||
|
|
||||||
|
def to_dict(self): |
||||||
|
return {"lang": "fr", "msg": self.msg, "type": "msg"} |
||||||
|
|
||||||
|
|
||||||
|
class BotMessage(AbstractResponse): |
||||||
|
|
||||||
|
def __init__(self, msg: str): |
||||||
|
self.msg = msg |
||||||
|
|
||||||
|
def to_dict(self): |
||||||
|
return {"type": "bot", "msg": self.msg} |
||||||
|
|
||||||
|
|
||||||
|
class Sound(AbstractResponse): |
||||||
|
|
||||||
|
def __init__(self, sound_filepath: str): |
||||||
|
self.filepath = sound_filepath |
||||||
|
|
||||||
|
def get_bytes(self): |
||||||
|
with open(self.filepath, "rb") as soundfile: |
||||||
|
return soundfile.read() |
||||||
|
|
||||||
|
|
||||||
|
class AttackCommand(AbstractResponse): |
||||||
|
|
||||||
|
def __init__(self, target_name: str, offset=1): |
||||||
|
self.target = target_name |
||||||
|
self.offset = offset |
||||||
|
|
||||||
|
def to_dict(self): |
||||||
|
return {"target": self.target, "order": self.offset, "type": "attack"} |
||||||
|
|
||||||
|
|
||||||
|
class Sleep(AbstractResponse): |
||||||
|
|
||||||
|
def __init__(self, duration : int): |
||||||
|
self.duration = duration |
||||||
@ -0,0 +1,2 @@ |
|||||||
|
DEFAULT_COOKIE = "nwoiw" |
||||||
|
DEFAULT_CHANNEL = "zizi" |
||||||
@ -0,0 +1,3 @@ |
|||||||
|
from .messages import * |
||||||
|
from .connections import * |
||||||
|
from .commons import MessageDispatcher |
||||||
@ -0,0 +1,36 @@ |
|||||||
|
import logging |
||||||
|
from typing import List |
||||||
|
|
||||||
|
from tools.commons import UserList |
||||||
|
|
||||||
|
|
||||||
|
class MessageProcessor: |
||||||
|
"""Parent class for all processors. A processor is basically a bot response triggered by a condition. |
||||||
|
The match method returns a boolean telling if the trigger condition is verified, and the process |
||||||
|
method returns a string of the output message. All bot processors are to be grouped in a list then |
||||||
|
given to a Dispatcher""" |
||||||
|
|
||||||
|
def match(self, text : str, sender_id : str, users_list : UserList) -> bool: |
||||||
|
"""Returns true if this is the kind of text a processor should respond to""" |
||||||
|
pass |
||||||
|
|
||||||
|
def process(self, text : str, sender_id : str, users_list : UserList) -> str: |
||||||
|
"""Processes a message and returns an answer""" |
||||||
|
pass |
||||||
|
|
||||||
|
|
||||||
|
class MessageDispatcher: |
||||||
|
"""Dispatches the current input message to the first botprocessor that matches the context given |
||||||
|
to the dispatch method.""" |
||||||
|
|
||||||
|
def __init__(self, processor_list : List[MessageProcessor]): |
||||||
|
self.processor_list = processor_list |
||||||
|
|
||||||
|
def dispatch(self,text : str, sender_id : str, users_list : UserList) -> str: |
||||||
|
"""Tells its first botprocessor to match the message to process this message and returns its answer""" |
||||||
|
for processor in self.processor_list: |
||||||
|
if processor.match(text, sender_id, users_list): |
||||||
|
logging.info("Matched %s" % processor.__class__.__name__) |
||||||
|
return processor.process(text, sender_id, users_list) |
||||||
|
|
||||||
|
return None |
||||||
@ -0,0 +1,29 @@ |
|||||||
|
import logging |
||||||
|
from typing import List |
||||||
|
|
||||||
|
from tools.commons import UserList |
||||||
|
|
||||||
|
|
||||||
|
class ConnectProcessor: |
||||||
|
def match(self, sender_id: str, users_list: UserList) -> bool: |
||||||
|
"""Returns true if this is the kind of user connection a processor should respond to""" |
||||||
|
pass |
||||||
|
|
||||||
|
def process(self, sender_id: str, users_list: UserList) -> str: |
||||||
|
"""Processes a message and returns an answer""" |
||||||
|
pass |
||||||
|
|
||||||
|
|
||||||
|
class ConnectionDispatcher: |
||||||
|
|
||||||
|
def __init__(self, processor_list: List[ConnectProcessor]): |
||||||
|
self.processor_list = processor_list |
||||||
|
|
||||||
|
def dispatch(self, sender_id: str, users_list: UserList) -> str: |
||||||
|
"""Tells its first botprocessor to match the message to process this message and returns its answer""" |
||||||
|
for processor in self.processor_list: |
||||||
|
if processor.match(sender_id, users_list): |
||||||
|
logging.info("Matched %s" % processor.__class__.__name__) |
||||||
|
return processor.process(sender_id, users_list) |
||||||
|
|
||||||
|
return None |
||||||
@ -0,0 +1,33 @@ |
|||||||
|
from tools.commons import Message |
||||||
|
from .commons import * |
||||||
|
|
||||||
|
|
||||||
|
class DispatcherBotProcessor(MessageProcessor): |
||||||
|
"""A processor that matches a context, then forwards the message to a list of sub-processors. |
||||||
|
This enables the botprocessor-matching mechanism to behave kinda like a decision tree""" |
||||||
|
|
||||||
|
def __init__(self, processors_list : List[MessageProcessor]): |
||||||
|
self.dispatcher = MessageDispatcher(processors_list) |
||||||
|
|
||||||
|
def process(self, text : str, sender_id : str, users_list : UserList): |
||||||
|
return self.dispatcher.dispatch(text, sender_id, users_list) |
||||||
|
|
||||||
|
|
||||||
|
class CommandsDispatcherProcessor(DispatcherBotProcessor): |
||||||
|
"""Reacts to commands of the form '/botname command' or 'botname, command' """ |
||||||
|
|
||||||
|
def __init__(self, processors_list: List[MessageProcessor], trigger_word: str = None, default_response :str = None): |
||||||
|
super().__init__(processors_list) |
||||||
|
self.trigger = trigger_word |
||||||
|
self.default_response = default_response if default_response is not None else "Commande non reconnue, pd" |
||||||
|
|
||||||
|
def match(self, text : str, sender_id : str, users_list : UserList): |
||||||
|
trigger = self.trigger.upper() if self.trigger is not None else users_list.my_name.upper() |
||||||
|
return text.upper().startswith(trigger + ",") \ |
||||||
|
or text.upper().startswith("/" + trigger) |
||||||
|
|
||||||
|
def process(self, text : str, sender_id : str, users_list : UserList): |
||||||
|
without_cmd = text[len(users_list.my_name)+1:] |
||||||
|
response = super().process(without_cmd, sender_id, users_list) |
||||||
|
return Message(self.default_response) if response is None else response |
||||||
|
|
||||||
Loading…
Reference in new issue