1
0
mirror of http://aero2k.de/t/repos/urlbot-native.git synced 2017-09-06 15:25:38 +02:00
Files
urlbot-native-trex/urlbot.py

330 lines
11 KiB
Python
Raw Normal View History

2015-12-20 15:24:42 +01:00
#!/usr/bin/env python3
2014-08-10 22:10:00 +02:00
# -*- coding: utf-8 -*-
2015-11-30 19:50:11 +01:00
"""
The URLBot - ready to strive for desaster in YOUR jabber MUC
"""
import sys
import time
from common import (
2015-11-30 19:17:40 +01:00
rate_limit_classes,
RATE_GLOBAL,
RATE_CHAT,
RATE_EVENT,
rate_limit,
2015-12-26 13:50:21 +01:00
ptypes_PARSE, ptypes_COMMAND)
2015-12-20 15:24:42 +01:00
from config import runtimeconf_set
from idlebot import IdleBot, start
2015-11-20 21:56:37 +01:00
from plugins import (
2015-11-30 19:17:40 +01:00
plugins as plugin_storage,
ptypes_COMMAND,
plugin_enabled_get,
ptypes_PARSE,
register_event,
else_command
2015-11-20 21:56:37 +01:00
)
import config
2015-07-19 20:47:57 +02:00
class UrlBot(IdleBot):
2015-11-30 19:50:11 +01:00
"""
The URLBot, doing things the IdleBot wouldn't dare to.
"""
2015-11-30 19:17:40 +01:00
def __init__(self, jid, password, rooms, nick):
super(UrlBot, self).__init__(jid, password, rooms, nick)
self.hist_ts = {p: [] for p in rate_limit_classes}
self.hist_flag = {p: True for p in rate_limit_classes}
self.add_event_handler('message', self.message)
self.priority = 100
2015-11-30 19:50:11 +01:00
for room in self.rooms:
self.add_event_handler('muc::%s::got_online' % room, self.muc_online)
2015-11-30 19:17:40 +01:00
def muc_message(self, msg_obj):
2015-11-30 19:50:11 +01:00
"""
Handle muc messages, return if irrelevant content or die by hangup.
:param msg_obj:
:return:
"""
2015-11-30 19:17:40 +01:00
return super(UrlBot, self).muc_message(msg_obj) and self.handle_msg(msg_obj)
def message(self, msg_obj):
2015-11-30 19:50:11 +01:00
"""
General message hook
:param msg_obj:
"""
if msg_obj['type'] == 'groupchat':
self.logger.info("Got the following MUC message: %s", str(msg_obj))
2015-11-30 19:17:40 +01:00
return
else:
2015-11-30 19:50:11 +01:00
self.logger.info("Got the following PM: %s", str(msg_obj))
self.handle_msg(msg_obj)
2015-11-30 19:17:40 +01:00
def muc_online(self, msg_obj):
"""
Hook for muc event "user joins"
"""
# don't react to yourself
if msg_obj['muc']['nick'] == self.nick:
return
# TODO: move this to a undirected plugin, maybe new plugin type
arg_user = msg_obj['muc']['nick']
arg_user_key = arg_user.lower()
user_records = config.runtimeconf_get('user_records')
2015-11-30 19:17:40 +01:00
2015-12-20 15:24:42 +01:00
if arg_user_key in user_records:
records = user_records[arg_user_key]
2015-11-30 19:17:40 +01:00
if not records:
return
self.send_message(
mto=msg_obj['from'].bare,
mbody='%s, there %s %d message%s for you:\n%s' % (
arg_user,
2015-11-30 19:50:11 +01:00
'is' if len(records) == 1 else 'are',
2015-11-30 19:17:40 +01:00
len(records),
2015-11-30 19:50:11 +01:00
'' if len(records) == 1 else 's',
2015-11-30 19:17:40 +01:00
'\n'.join(records)
),
mtype='groupchat'
)
2015-11-30 19:50:11 +01:00
self.logger.info('sent %d offline records to room %s',
len(records), msg_obj['from'].bare)
2015-11-30 19:17:40 +01:00
2015-12-20 15:24:42 +01:00
if config.conf_get('persistent_locked'):
2015-11-30 19:50:11 +01:00
self.logger.warning("couldn't get exclusive lock")
2015-11-30 19:17:40 +01:00
return False
2015-12-20 15:24:42 +01:00
config.conf_set('persistent_locked', True)
2015-11-30 19:17:40 +01:00
2015-12-21 09:52:01 +01:00
user_records.pop(arg_user_key)
config.runtimeconf_persist()
2015-11-30 19:17:40 +01:00
2015-12-20 15:24:42 +01:00
config.conf_set('persistent_locked', False)
2015-11-30 19:17:40 +01:00
# @rate_limited(10)
def send_reply(self, message, msg_obj=None):
"""
Send a reply to a message
"""
if self.show:
2015-11-30 19:50:11 +01:00
self.logger.warning("I'm muted! (status: %s)", self.show)
2015-11-30 19:17:40 +01:00
return
request_counter = int(config.runtimeconf_get('request_counter'))
config.runtimeconf_set('request_counter', request_counter + 1)
2015-11-30 19:17:40 +01:00
if str is not type(message):
message = '\n'.join(message)
def cached(function, ttl=60):
cache = {}
ttl = 60
now = time.time()
def wrapper(*args):
hash_ = hash(args)
if hash_ in cache and cache[args]['time'] < now - ttl:
return cache[hash_]['result']
else:
result = function(*args)
cache[hash_] = {}
cache[hash_]['time'] = now
cache[hash_]['result'] = result
return result
return wrapper
@cached
def get_bots_present(room):
other_bots = config.runtimeconf_get("other_bots")
2015-12-20 15:24:42 +01:00
if not other_bots:
return False
users = self.plugin['xep_0045'].getRoster(room)
return set(users).intersection(set(other_bots))
2015-11-30 19:17:40 +01:00
def _prevent_panic(message, room):
"""check other bots, add nospoiler with urls"""
2015-12-20 15:24:42 +01:00
if 'http' in message and get_bots_present(room):
message = '(nospoiler) %s' % message
2015-11-30 19:17:40 +01:00
return message
2015-12-20 15:24:42 +01:00
if config.conf_get('debug_mode'):
2015-11-30 19:17:40 +01:00
print(message)
else:
if msg_obj:
# TODO: bot modes off/on/auto... this should be active for "on".
# message = _prevent_panic(message, msg_obj['from'].bare)
# if get_bots_present(msg_obj['from'].bare):
# return
if msg_obj['type'] == 'groupchat':
if msg_obj['mucnick'] in config.runtimeconf_get("other_bots", ()):
msg_obj['type'] = 'chat'
self.send_reply("You're flagged as bot, please write {}: remove-from-botlist "
"{} if you're not a bot.".format(
config.conf_get('bot_nickname'), msg_obj['mucnick']
), msg_obj)
self.logger.debug("not talking to the other bot named {}".format(msg_obj['mucnick']))
return False
self.send_message(
mto=msg_obj['from'].bare,
mbody=message,
mtype='groupchat'
)
elif msg_obj['type'] == 'chat':
self.send_message(
mto=msg_obj['from'],
mbody=message,
mtype='chat'
)
else:
2015-11-30 19:17:40 +01:00
for room in self.rooms:
# message = _prevent_panic(message, room)
if get_bots_present(room):
continue
2015-11-30 19:17:40 +01:00
self.send_message(
mto=room,
mbody=message,
mtype='groupchat'
)
def handle_msg(self, msg_obj):
"""
called for incoming messages
:param msg_obj:
:returns nothing
"""
content = msg_obj['body']
if 'has set the subject to:' in content:
return
if sys.argv[0] in content:
self.logger.info('silenced, this is my own log')
return
if 'nospoiler' in content:
self.logger.info('no spoiler for: ' + content)
return
if msg_obj['mucnick'] in config.runtime_config_store['spammers']:
self.logger.info("ignoring spammer {}".format(msg_obj['mucnick']))
return
2015-12-20 15:24:42 +01:00
try:
self.data_parse_commands(msg_obj)
self.data_parse_other(msg_obj)
except Exception as e:
self.logger.exception(e)
2015-11-30 19:17:40 +01:00
def data_parse_commands(self, msg_obj):
"""
react to a message with the bots nick
:param msg_obj: dictionary with incoming message parameters
:returns: nothing
"""
data = msg_obj['body']
words = data.split()
2015-11-30 19:50:11 +01:00
if len(words) < 2: # need at least two words
2015-11-30 19:17:40 +01:00
return None
# don't reply if beginning of the text matches bot_nickname
2015-12-20 15:24:42 +01:00
if not data.startswith(config.conf_get('bot_nickname')):
2015-11-30 19:17:40 +01:00
return None
if 'hangup' in data:
2015-11-30 19:50:11 +01:00
self.logger.warning('received hangup: ' + data)
self.hangup()
2015-11-30 19:17:40 +01:00
sys.exit(1)
reply_user = msg_obj['mucnick'] or msg_obj['from']._jid[2]
2015-11-30 19:17:40 +01:00
2015-11-30 19:50:11 +01:00
# TODO: check how several commands/plugins
# in a single message behave (also with rate limiting)
2015-11-30 19:17:40 +01:00
reacted = False
2015-12-20 21:15:16 +01:00
for plugin in filter(lambda p: p.plugin_name == words[1], plugin_storage[ptypes_COMMAND]):
2015-11-30 19:17:40 +01:00
if not plugin_enabled_get(plugin):
continue
ret = plugin(
data=data,
cmd_list=[pl.plugin_name for pl in plugin_storage[ptypes_COMMAND]],
parser_list=[pl.plugin_name for pl in plugin_storage[ptypes_PARSE]],
reply_user=reply_user,
msg_obj=msg_obj,
argv=words[1:]
)
if ret:
self._run_action(ret, plugin, msg_obj)
reacted = True
if not reacted and rate_limit(RATE_GLOBAL):
ret = else_command({'reply_user': reply_user})
if ret:
if 'msg' in ret:
self.send_reply(ret['msg'], msg_obj)
def data_parse_other(self, msg_obj):
"""
react to any message
:param msg_obj: incoming message parameters
:return:
"""
data = msg_obj['body']
reply_user = msg_obj['mucnick']
for plugin in plugin_storage[ptypes_PARSE]:
if not plugin_enabled_get(plugin):
continue
ret = plugin(reply_user=reply_user, data=data)
if ret:
self._run_action(ret, plugin, msg_obj)
def _run_action(self, action, plugin, msg_obj):
"""
Execute the plugin's execution plan
:param action: dict with event and/or msg
:param plugin: plugin obj
:param msg_obj: xmpp message obj
"""
if 'event' in action:
event = action["event"]
if 'msg' in event:
register_event(event["time"], self.send_reply, [event['msg'], msg_obj])
2015-11-30 19:17:40 +01:00
elif 'command' in event:
command = event["command"]
if rate_limit(RATE_EVENT):
register_event(event["time"], command[0], command[1])
if 'msg' in action and rate_limit(RATE_CHAT | plugin.ratelimit_class):
self.send_reply(action['msg'], msg_obj)
if 'priv_msg' in action and rate_limit(RATE_CHAT | plugin.ratelimit_class):
msg_obj['type'] = 'chat'
self.send_reply(action['priv_msg'], msg_obj)
2015-11-30 19:17:40 +01:00
if 'presence' in action:
presence = action['presence']
2015-12-20 15:24:42 +01:00
runtimeconf_set('presence', presence)
2015-11-30 19:17:40 +01:00
self.status = presence.get('msg')
self.show = presence.get('status')
self.send_presence(pstatus=self.status, pshow=self.show)
# self.reconnect(wait=True)
2015-11-30 19:17:40 +01:00
2015-11-30 19:50:11 +01:00
if __name__ == '__main__':
2015-11-30 19:17:40 +01:00
start(UrlBot, True)