mirror of
https://github.com/mightyBroccoli/xmpp-chatbot.git
synced 2024-12-04 22:33:36 +01:00
new async logic
This commit is contained in:
parent
d305f8adf3
commit
31ddc8aeb4
2 changed files with 91 additions and 25 deletions
59
common/misc.py
Executable file
59
common/misc.py
Executable file
|
@ -0,0 +1,59 @@
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
import validators
|
||||||
|
from common.strings import StaticAnswers
|
||||||
|
|
||||||
|
|
||||||
|
def deduplicate(reply):
|
||||||
|
"""
|
||||||
|
list deduplication method
|
||||||
|
:param list reply: list containing non unique items
|
||||||
|
:return: list containing unique items
|
||||||
|
"""
|
||||||
|
reply_dedup = list()
|
||||||
|
for item in reply:
|
||||||
|
if item not in reply_dedup:
|
||||||
|
reply_dedup.append(item)
|
||||||
|
|
||||||
|
return reply_dedup
|
||||||
|
|
||||||
|
|
||||||
|
def validate(wordlist, index):
|
||||||
|
"""
|
||||||
|
validation method to reduce malformed querys and unnecessary connection attempts
|
||||||
|
:param wordlist: words separated by " " from the message
|
||||||
|
:param index: keyword index inside the message
|
||||||
|
:return: true if valid
|
||||||
|
"""
|
||||||
|
# keyword inside the message
|
||||||
|
argument = wordlist[index]
|
||||||
|
|
||||||
|
# check if argument is in the argument list
|
||||||
|
if argument in StaticAnswers().keys(arg='list'):
|
||||||
|
# if argument uses a domain check for occurrence in list and check domain
|
||||||
|
if argument in StaticAnswers().keys(arg='list', keyword='domain_keywords'):
|
||||||
|
try:
|
||||||
|
target = wordlist[index + 1]
|
||||||
|
if validators.domain(target):
|
||||||
|
return True
|
||||||
|
elif validators.email(target):
|
||||||
|
return True
|
||||||
|
|
||||||
|
except IndexError:
|
||||||
|
# except an IndexError if a keywords is the last word in the message
|
||||||
|
return False
|
||||||
|
|
||||||
|
# check if number keyword is used if true check if target is assignable
|
||||||
|
elif argument in StaticAnswers().keys(arg='list', keyword='number_keywords'):
|
||||||
|
try:
|
||||||
|
if wordlist[index + 1]:
|
||||||
|
return True
|
||||||
|
except IndexError:
|
||||||
|
# except an IndexError if target is not assignable
|
||||||
|
return False
|
||||||
|
# check if argument is inside no_arg list
|
||||||
|
elif argument in StaticAnswers().keys(arg='list', keyword="no_arg_keywords"):
|
||||||
|
return True
|
||||||
|
else:
|
||||||
|
return False
|
||||||
|
else:
|
||||||
|
return False
|
57
main.py
57
main.py
|
@ -17,6 +17,7 @@ import logging
|
||||||
from argparse import ArgumentParser
|
from argparse import ArgumentParser
|
||||||
from slixmpp.exceptions import XMPPError
|
from slixmpp.exceptions import XMPPError
|
||||||
|
|
||||||
|
import common.misc as misc
|
||||||
from common.strings import StaticAnswers
|
from common.strings import StaticAnswers
|
||||||
from classes.functions import Version, LastActivity, ContactInfo, HandleError
|
from classes.functions import Version, LastActivity, ContactInfo, HandleError
|
||||||
from classes.xep import XEPRequest
|
from classes.xep import XEPRequest
|
||||||
|
@ -29,10 +30,16 @@ class QueryBot(slixmpp.ClientXMPP):
|
||||||
self.room = room
|
self.room = room
|
||||||
self.nick = nick
|
self.nick = nick
|
||||||
|
|
||||||
|
self.data = {
|
||||||
|
'words': list(),
|
||||||
|
'reply': list(),
|
||||||
|
'queue': list()
|
||||||
|
}
|
||||||
|
|
||||||
# session start event, starting point for the presence and roster requests
|
# session start event, starting point for the presence and roster requests
|
||||||
self.add_event_handler('session_start', self.start)
|
self.add_event_handler('session_start', self.start)
|
||||||
|
|
||||||
# register handler to recieve both groupchat and normal message events
|
# register recieve handler for both groupchat and normal message events
|
||||||
self.add_event_handler('message', self.message)
|
self.add_event_handler('message', self.message)
|
||||||
|
|
||||||
def start(self, event):
|
def start(self, event):
|
||||||
|
@ -97,68 +104,68 @@ class QueryBot(slixmpp.ClientXMPP):
|
||||||
|
|
||||||
return reply_dedup
|
return reply_dedup
|
||||||
|
|
||||||
@asyncio.coroutine
|
async def message(self, msg):
|
||||||
def message(self, msg):
|
|
||||||
"""
|
"""
|
||||||
:param msg: received message stanza
|
:param msg: received message stanza
|
||||||
"""
|
"""
|
||||||
# init empty reply list
|
data = self.data
|
||||||
reply = list()
|
|
||||||
|
|
||||||
# catch self messages to prevent self flooding
|
# catch self messages to prevent self flooding
|
||||||
if msg['mucnick'] == self.nick:
|
if msg['mucnick'] == self.nick:
|
||||||
return
|
return
|
||||||
|
|
||||||
elif self.nick in msg['body']:
|
elif self.nick in msg['body']:
|
||||||
# add pre predefined text to reply list
|
# add pre predefined text to reply list
|
||||||
reply.append(StaticAnswers(msg['mucnick']).gen_answer())
|
data['reply'].append(StaticAnswers(msg['mucnick']).gen_answer())
|
||||||
|
|
||||||
# building the queue
|
# building the queue
|
||||||
# double splitting to exclude whitespaces
|
# double splitting to exclude whitespaces
|
||||||
words = " ".join(msg['body'].split()).split(sep=" ")
|
data['words'] = " ".join(msg['body'].split()).split(sep=" ")
|
||||||
queue = list()
|
|
||||||
|
|
||||||
# check all words in side the message for possible hits
|
# check all words in side the message for possible hits
|
||||||
for x in enumerate(words):
|
for x in enumerate(data['words']):
|
||||||
# check word for match in keywords list
|
# check word for match in keywords list
|
||||||
for y in StaticAnswers().keys(arg='list'):
|
for y in StaticAnswers().keys(arg='list'):
|
||||||
# if so queue the keyword and the postion in the string
|
# if so queue the keyword and the position in the string
|
||||||
if x[1] == y:
|
if x[1] == y:
|
||||||
# only add job to queue if domain is valid
|
# only add job to queue if domain is valid
|
||||||
if self.validate(words, x[0]):
|
if misc.validate(data['words'], x[0]):
|
||||||
queue.append({str(y): x[0]})
|
data['queue'].append({str(y): x[0]})
|
||||||
|
|
||||||
# queue
|
# queue
|
||||||
for job in queue:
|
for job in data['queue']:
|
||||||
for keyword in job:
|
for keyword in job:
|
||||||
index = job[keyword]
|
index = job[keyword]
|
||||||
|
|
||||||
if keyword == '!help':
|
if keyword == '!help':
|
||||||
reply.append(StaticAnswers().gen_help())
|
data['reply'].append(StaticAnswers().gen_help())
|
||||||
continue
|
continue
|
||||||
|
|
||||||
target = words[index + 1]
|
target = data['words'][index + 1]
|
||||||
try:
|
try:
|
||||||
if keyword == '!uptime':
|
if keyword == '!uptime':
|
||||||
last_activity = yield from self['xep_0012'].get_last_activity(jid=target, cached=False)
|
last_activity = await self['xep_0012'].get_last_activity(jid=target)
|
||||||
reply.append(LastActivity(last_activity, msg, target).format_values())
|
self.data['reply'].append(LastActivity(last_activity, msg, target).format_values())
|
||||||
|
|
||||||
elif keyword == "!version":
|
elif keyword == "!version":
|
||||||
version = yield from self['xep_0092'].get_version(jid=target, cached=False)
|
version = await self['xep_0092'].get_version(jid=target)
|
||||||
reply.append(Version(version, msg, target).format_version())
|
self.data['reply'].append(Version(version, msg, target).format_version())
|
||||||
|
|
||||||
|
|
||||||
elif keyword == "!contact":
|
elif keyword == "!contact":
|
||||||
contact = yield from self['xep_0030'].get_info(jid=target, cached=False)
|
last_activity = await self['xep_0012'].get_last_activity(jid=target)
|
||||||
reply.append(ContactInfo(contact, msg, target).format_contact())
|
self.data['reply'].append(LastActivity(last_activity, msg, target).format_values())
|
||||||
|
|
||||||
|
|
||||||
elif keyword == "!xep":
|
elif keyword == "!xep":
|
||||||
reply.append(XEPRequest(msg, target).format())
|
data['reply'].append(XEPRequest(msg, target).format())
|
||||||
|
|
||||||
except XMPPError as error:
|
except XMPPError as error:
|
||||||
reply.append(HandleError(error, msg, keyword, target).build_report())
|
data['reply'].append(HandleError(error, msg, keyword, target).build_report())
|
||||||
|
|
||||||
# remove None type from list and send all elements
|
# remove None type from list and send all elements
|
||||||
if list(filter(None.__ne__, reply)) and reply:
|
if list(filter(None.__ne__, data['reply'])) and data['reply']:
|
||||||
reply = self.deduplicate(reply)
|
reply = misc.deduplicate(data['reply'])
|
||||||
self.send_message(mto=msg['from'].bare, mbody="\n".join(reply), mtype=msg['type'])
|
self.send_message(mto=msg['from'].bare, mbody="\n".join(reply), mtype=msg['type'])
|
||||||
|
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue