@@ -2,7 +2,7 @@ | |||||
*.pyc | *.pyc | ||||
# Ignore bot-specific config file: | # Ignore bot-specific config file: | ||||
config.json | |||||
config.yml | |||||
# Ignore logs directory: | # Ignore logs directory: | ||||
logs/ | logs/ | ||||
@@ -32,5 +32,5 @@ __version__ = "0.1.dev" | |||||
__email__ = "ben.kurtovic@verizon.net" | __email__ = "ben.kurtovic@verizon.net" | ||||
from earwigbot import ( | from earwigbot import ( | ||||
blowfish, commands, config, irc, main, rules, runner, tasks, tests, wiki | |||||
blowfish, commands, config, irc, main, runner, tasks, tests, wiki | |||||
) | ) |
@@ -21,7 +21,7 @@ | |||||
# SOFTWARE. | # SOFTWARE. | ||||
""" | """ | ||||
EarwigBot's JSON Config File Parser | |||||
EarwigBot's YAML Config File Parser | |||||
This handles all tasks involving reading and writing to our config file, | This handles all tasks involving reading and writing to our config file, | ||||
including encrypting and decrypting passwords and making a new config file from | including encrypting and decrypting passwords and making a new config file from | ||||
@@ -45,11 +45,12 @@ Additionally, _BotConfig has some functions used in config loading: | |||||
variables; won't work if passwords aren't encrypted | variables; won't work if passwords aren't encrypted | ||||
""" | """ | ||||
import json | |||||
import logging | import logging | ||||
import logging.handlers | import logging.handlers | ||||
from os import mkdir, path | from os import mkdir, path | ||||
import yaml | |||||
from earwigbot import blowfish | from earwigbot import blowfish | ||||
__all__ = ["config"] | __all__ = ["config"] | ||||
@@ -90,7 +91,7 @@ class _BotConfig(object): | |||||
def __init__(self): | def __init__(self): | ||||
self._script_dir = path.dirname(path.abspath(__file__)) | self._script_dir = path.dirname(path.abspath(__file__)) | ||||
self._root_dir = path.split(self._script_dir)[0] | self._root_dir = path.split(self._script_dir)[0] | ||||
self._config_path = path.join(self._root_dir, "config.json") | |||||
self._config_path = path.join(self._root_dir, "config.yml") | |||||
self._log_dir = path.join(self._root_dir, "logs") | self._log_dir = path.join(self._root_dir, "logs") | ||||
self._decryption_key = None | self._decryption_key = None | ||||
self._data = None | self._data = None | ||||
@@ -105,12 +106,12 @@ class _BotConfig(object): | |||||
self._metadata] | self._metadata] | ||||
def _load(self): | def _load(self): | ||||
"""Load data from our JSON config file (config.json) into _config.""" | |||||
"""Load data from our JSON config file (config.yml) into _config.""" | |||||
filename = self._config_path | filename = self._config_path | ||||
with open(filename, 'r') as fp: | with open(filename, 'r') as fp: | ||||
try: | try: | ||||
self._data = json.load(fp) | |||||
except ValueError as error: | |||||
self._data = yaml.load(fp) | |||||
except yaml.YAMLError as error: | |||||
print "Error parsing config file {0}:".format(filename) | print "Error parsing config file {0}:".format(filename) | ||||
print error | print error | ||||
exit(1) | exit(1) | ||||
@@ -158,7 +159,7 @@ class _BotConfig(object): | |||||
def _make_new(self): | def _make_new(self): | ||||
"""Make a new config file based on the user's input.""" | """Make a new config file based on the user's input.""" | ||||
encrypt = raw_input("Would you like to encrypt passwords stored in config.json? [y/n] ") | |||||
encrypt = raw_input("Would you like to encrypt passwords stored in config.yml? [y/n] ") | |||||
if encrypt.lower().startswith("y"): | if encrypt.lower().startswith("y"): | ||||
is_encrypted = True | is_encrypted = True | ||||
else: | else: | ||||
@@ -181,6 +182,11 @@ class _BotConfig(object): | |||||
@property | @property | ||||
def log_dir(self): | def log_dir(self): | ||||
return self._log_dir | return self._log_dir | ||||
@property | |||||
def data(self): | |||||
"""The entire config file.""" | |||||
return self._data | |||||
@property | @property | ||||
def components(self): | def components(self): | ||||
@@ -22,7 +22,6 @@ | |||||
import logging | import logging | ||||
from earwigbot import rules | |||||
from earwigbot.irc import IRCConnection, RC, BrokenSocketException | from earwigbot.irc import IRCConnection, RC, BrokenSocketException | ||||
from earwigbot.config import config | from earwigbot.config import config | ||||
@@ -34,7 +33,7 @@ class Watcher(IRCConnection): | |||||
The IRC watcher runs on a wiki recent-changes server and listens for | The IRC watcher runs on a wiki recent-changes server and listens for | ||||
edits. Users cannot interact with this part of the bot. When an event | edits. Users cannot interact with this part of the bot. When an event | ||||
occurs, we run it through rules.py's process() function, which can result | |||||
occurs, we run it through some rules stored in our config, which can result | |||||
in wiki bot tasks being started (located in tasks/) or messages being sent | in wiki bot tasks being started (located in tasks/) or messages being sent | ||||
to channels on the IRC frontend. | to channels on the IRC frontend. | ||||
""" | """ | ||||
@@ -46,6 +45,7 @@ class Watcher(IRCConnection): | |||||
base.__init__(cf["host"], cf["port"], cf["nick"], cf["ident"], | base.__init__(cf["host"], cf["port"], cf["nick"], cf["ident"], | ||||
cf["realname"], self.logger) | cf["realname"], self.logger) | ||||
self.frontend = frontend | self.frontend = frontend | ||||
self._prepare_process_hook() | |||||
self._connect() | self._connect() | ||||
def _process_message(self, line): | def _process_message(self, line): | ||||
@@ -63,7 +63,7 @@ class Watcher(IRCConnection): | |||||
msg = " ".join(line[3:])[1:] | msg = " ".join(line[3:])[1:] | ||||
rc = RC(msg) # New RC object to store this event's data | rc = RC(msg) # New RC object to store this event's data | ||||
rc.parse() # Parse a message into pagenames, usernames, etc. | rc.parse() # Parse a message into pagenames, usernames, etc. | ||||
self._process_rc(rc) # Report to frontend channels or start tasks | |||||
self._process_rc_event(rc) | |||||
# If we are pinged, pong back: | # If we are pinged, pong back: | ||||
elif line[0] == "PING": | elif line[0] == "PING": | ||||
@@ -74,14 +74,40 @@ class Watcher(IRCConnection): | |||||
for chan in config.irc["watcher"]["channels"]: | for chan in config.irc["watcher"]["channels"]: | ||||
self.join(chan) | self.join(chan) | ||||
def _process_rc(self, rc): | |||||
def _prepare_process_hook(self): | |||||
"""Create our RC event process hook from information in config. | |||||
This will get put in the function self._process_hook, which takes an RC | |||||
object and returns a list of frontend channels to report this event to. | |||||
""" | |||||
# Default RC process hook does nothing: | |||||
self._process_hook = lambda rc: () | |||||
try: | |||||
rules = config.data["rules"] | |||||
except KeyError: | |||||
return | |||||
try: | |||||
module = compile(rules, config.config_path, "exec") | |||||
except Exception: | |||||
e = "Could not compile config file's RC event rules" | |||||
self.logger.exception(e) | |||||
return | |||||
try: | |||||
self._process_hook = module.process | |||||
except AttributeError: | |||||
e = "RC event rules compiled correctly, but no process(rc) function was found" | |||||
self.logger.error(e) | |||||
return | |||||
def _process_rc_event(self, rc): | |||||
"""Process a recent change event from IRC (or, an RC object). | """Process a recent change event from IRC (or, an RC object). | ||||
The actual processing is configurable, so we don't have that hard-coded | The actual processing is configurable, so we don't have that hard-coded | ||||
here. We simply call rules's process() function and expect a list of | |||||
channels back, which we report the event data to. | |||||
here. We simply call our process hook (self._process_hook), created by | |||||
self._prepare_process_hook() from information in the "rules" section of | |||||
our config. | |||||
""" | """ | ||||
chans = rules.process(rc) | |||||
chans = self._process_hook(rc) | |||||
if chans and self.frontend: | if chans and self.frontend: | ||||
pretty = rc.prettify() | pretty = rc.prettify() | ||||
for chan in chans: | for chan in chans: | ||||
@@ -1,85 +0,0 @@ | |||||
# -*- coding: utf-8 -*- | |||||
# | |||||
# Copyright (C) 2009-2012 by Ben Kurtovic <ben.kurtovic@verizon.net> | |||||
# | |||||
# Permission is hereby granted, free of charge, to any person obtaining a copy | |||||
# of this software and associated documentation files (the "Software"), to deal | |||||
# in the Software without restriction, including without limitation the rights | |||||
# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | |||||
# copies of the Software, and to permit persons to whom the Software is | |||||
# furnished to do so, subject to the following conditions: | |||||
# | |||||
# The above copyright notice and this permission notice shall be included in | |||||
# all copies or substantial portions of the Software. | |||||
# | |||||
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | |||||
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | |||||
# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | |||||
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | |||||
# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | |||||
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | |||||
# SOFTWARE. | |||||
""" | |||||
EarwigBot's IRC Watcher Rules | |||||
This file contains (configurable!) rules that EarwigBot's watcher uses after it | |||||
recieves an event from IRC. | |||||
""" | |||||
import re | |||||
from earwigbot.tasks import task_manager | |||||
afc_prefix = "wikipedia( talk)?:(wikiproject )?articles for creation" | |||||
# compile some regexps used when finding specific events | |||||
r_page = re.compile(afc_prefix) | |||||
r_ffu = re.compile("wikipedia( talk)?:files for upload") | |||||
r_move1 = re.compile("moved \[\[{}".format(afc_prefix)) | |||||
r_move2 = re.compile("moved \[\[(.*?)\]\] to \[\[{}".format(afc_prefix)) | |||||
r_moved_pages = re.compile("^moved \[\[(.*?)\]\] to \[\[(.*?)\]\]") | |||||
r_delete = re.compile("deleted \"\[\[{}".format(afc_prefix)) | |||||
r_deleted_page = re.compile("^deleted \"\[\[(.*?)\]\]") | |||||
r_restore = re.compile("restored \"\[\[{}".format(afc_prefix)) | |||||
r_restored_page = re.compile("^restored \"\[\[(.*?)\]\]") | |||||
r_protect = re.compile("protected \"\[\[{}".format(afc_prefix)) | |||||
def process(rc): | |||||
"""Given an RC() object, return a list of channels to report this event to. | |||||
Also, start any wiki bot tasks within this function if necessary.""" | |||||
chans = set() # channels to report this message to | |||||
page_name = rc.page.lower() | |||||
comment = rc.comment.lower() | |||||
if "!earwigbot" in rc.msg.lower(): | |||||
chans.update(("##earwigbot", "#wikipedia-en-afc-feed")) | |||||
if r_page.search(page_name): | |||||
#task_manager.start("afc_copyvios", page=rc.page) | |||||
chans.add("#wikipedia-en-afc-feed") | |||||
elif r_ffu.match(page_name): | |||||
chans.add("#wikipedia-en-afc-feed") | |||||
elif page_name.startswith("template:afc submission"): | |||||
chans.add("#wikipedia-en-afc-feed") | |||||
elif rc.flags == "move" and (r_move1.match(comment) or | |||||
r_move2.match(comment)): | |||||
p = r_moved_pages.findall(rc.comment)[0] | |||||
chans.add("#wikipedia-en-afc-feed") | |||||
elif rc.flags == "delete" and r_delete.match(comment): | |||||
p = r_deleted_page.findall(rc.comment)[0] | |||||
chans.add("#wikipedia-en-afc-feed") | |||||
elif rc.flags == "restore" and r_restore.match(comment): | |||||
p = r_restored_page.findall(rc.comment)[0] | |||||
#task_manager.start("afc_copyvios", page=p) | |||||
chans.add("#wikipedia-en-afc-feed") | |||||
elif rc.flags == "protect" and r_protect.match(comment): | |||||
chans.add("#wikipedia-en-afc-feed") | |||||
return chans |
@@ -44,7 +44,7 @@ def run(): | |||||
from earwigbot import main | from earwigbot import main | ||||
root_dir = raw_input() | root_dir = raw_input() | ||||
config_path = path.join(root_dir, "config.json") | |||||
config_path = path.join(root_dir, "config.yml") | |||||
log_dir = path.join(root_dir, "logs") | log_dir = path.join(root_dir, "logs") | ||||
is_encrypted = config.load(config_path, log_dir) | is_encrypted = config.load(config_path, log_dir) | ||||
if is_encrypted: | if is_encrypted: | ||||
@@ -63,8 +63,8 @@ def _get_cookiejar(): | |||||
one is returned every time. | one is returned every time. | ||||
The .cookies file is located in the project root, same directory as | The .cookies file is located in the project root, same directory as | ||||
config.json and earwigbot.py. If it doesn't exist, we will create the file | |||||
and set it to be readable and writeable only by us. If it exists but the | |||||
config.yml and bot.py. If it doesn't exist, we will create the file and set | |||||
it to be readable and writeable only by us. If it exists but the | |||||
information inside is bogus, we will ignore it. | information inside is bogus, we will ignore it. | ||||
This is normally called by _get_site_object_from_dict() (in turn called by | This is normally called by _get_site_object_from_dict() (in turn called by | ||||
@@ -116,14 +116,6 @@ def _get_site_object_from_dict(name, d): | |||||
user_agent = user_agent.replace("$1", earwigbot.__version__) | user_agent = user_agent.replace("$1", earwigbot.__version__) | ||||
user_agent = user_agent.replace("$2", platform.python_version()) | user_agent = user_agent.replace("$2", platform.python_version()) | ||||
for key, value in namespaces.items(): # Convert string keys to integers | |||||
del namespaces[key] | |||||
try: | |||||
namespaces[int(key)] = value | |||||
except ValueError: # Data is broken, ignore it | |||||
namespaces = None | |||||
break | |||||
return Site(name=name, project=project, lang=lang, base_url=base_url, | return Site(name=name, project=project, lang=lang, base_url=base_url, | ||||
article_path=article_path, script_path=script_path, sql=sql, | article_path=article_path, script_path=script_path, sql=sql, | ||||
namespaces=namespaces, login=login, cookiejar=cookiejar, | namespaces=namespaces, login=login, cookiejar=cookiejar, | ||||