@@ -24,8 +24,11 @@ import hashlib | |||||
from Crypto.Cipher import Blowfish | from Crypto.Cipher import Blowfish | ||||
from earwigbot import importer | |||||
from earwigbot.commands import Command | from earwigbot.commands import Command | ||||
Blowfish = importer.new("Crypto.Cipher.Blowfish") | |||||
class Crypt(Command): | class Crypt(Command): | ||||
"""Provides hash functions with !hash (!hash list for supported algorithms) | """Provides hash functions with !hash (!hash list for supported algorithms) | ||||
and Blowfish encryption with !encrypt and !decrypt.""" | and Blowfish encryption with !encrypt and !decrypt.""" | ||||
@@ -66,7 +69,13 @@ class Crypt(Command): | |||||
self.reply(data, msg.format(data.command)) | self.reply(data, msg.format(data.command)) | ||||
return | return | ||||
cipher = Blowfish.new(hashlib.sha256(key).digest()) | |||||
try: | |||||
cipher = Blowfish.new(hashlib.sha256(key).digest()) | |||||
except ImportError: | |||||
msg = "This command requires the 'pycrypto' package: https://www.dlitz.net/software/pycrypto/" | |||||
self.reply(data, msg) | |||||
return | |||||
try: | try: | ||||
if data.command == "encrypt": | if data.command == "encrypt": | ||||
if len(text) % 8: | if len(text) % 8: | ||||
@@ -56,7 +56,7 @@ class Time(Command): | |||||
try: | try: | ||||
tzinfo = pytz.timezone(timezone) | tzinfo = pytz.timezone(timezone) | ||||
except ImportError: | except ImportError: | ||||
msg = "This command requires the 'pytz' module: http://pytz.sourceforge.net/" | |||||
msg = "This command requires the 'pytz' package: http://pytz.sourceforge.net/" | |||||
self.reply(data, msg) | self.reply(data, msg) | ||||
return | return | ||||
except pytz.exceptions.UnknownTimeZoneError: | except pytz.exceptions.UnknownTimeZoneError: | ||||
@@ -28,10 +28,9 @@ import logging.handlers | |||||
from os import mkdir, path | from os import mkdir, path | ||||
import stat | import stat | ||||
from Crypto.Cipher import Blowfish | |||||
import bcrypt | |||||
import yaml | import yaml | ||||
from earwigbot import importer | |||||
from earwigbot.config.formatter import BotFormatter | from earwigbot.config.formatter import BotFormatter | ||||
from earwigbot.config.node import ConfigNode | from earwigbot.config.node import ConfigNode | ||||
from earwigbot.config.ordered_yaml import OrderedLoader | from earwigbot.config.ordered_yaml import OrderedLoader | ||||
@@ -39,6 +38,9 @@ from earwigbot.config.permissions import PermissionsDB | |||||
from earwigbot.config.script import ConfigScript | from earwigbot.config.script import ConfigScript | ||||
from earwigbot.exceptions import NoConfigError | from earwigbot.exceptions import NoConfigError | ||||
Blowfish = importer.new("Crypto.Cipher.Blowfish") | |||||
bcrypt = importer.new("bcrypt") | |||||
__all__ = ["BotConfig"] | __all__ = ["BotConfig"] | ||||
class BotConfig(object): | class BotConfig(object): | ||||
@@ -29,13 +29,14 @@ import stat | |||||
import sys | import sys | ||||
from textwrap import fill, wrap | from textwrap import fill, wrap | ||||
from Crypto.Cipher import Blowfish | |||||
import bcrypt | |||||
import yaml | import yaml | ||||
from earwigbot import exceptions | |||||
from earwigbot import exceptions, importer | |||||
from earwigbot.config.ordered_yaml import OrderedDumper | from earwigbot.config.ordered_yaml import OrderedDumper | ||||
Blowfish = importer.new("Crypto.Cipher.Blowfish") | |||||
bcrypt = importer.new("bcrypt") | |||||
__all__ = ["ConfigScript"] | __all__ = ["ConfigScript"] | ||||
RULES_TEMPLATE = """# -*- coding: utf-8 -*- | RULES_TEMPLATE = """# -*- coding: utf-8 -*- | ||||
@@ -145,17 +146,30 @@ class ConfigScript(object): | |||||
is to run on a public computer like the Toolserver, but | is to run on a public computer like the Toolserver, but | ||||
otherwise the need to enter a key everytime you start | otherwise the need to enter a key everytime you start | ||||
the bot may be annoying.""") | the bot may be annoying.""") | ||||
self.data["metadata"]["encryptPasswords"] = False | |||||
if self._ask_bool("Encrypt stored passwords?"): | if self._ask_bool("Encrypt stored passwords?"): | ||||
self.data["metadata"]["encryptPasswords"] = True | |||||
key = getpass(self.PROMPT + "Enter an encryption key: ") | key = getpass(self.PROMPT + "Enter an encryption key: ") | ||||
msg = "Running {0} rounds of bcrypt...".format(self.BCRYPT_ROUNDS) | msg = "Running {0} rounds of bcrypt...".format(self.BCRYPT_ROUNDS) | ||||
self._print_no_nl(msg) | self._print_no_nl(msg) | ||||
signature = bcrypt.hashpw(key, bcrypt.gensalt(self.BCRYPT_ROUNDS)) | |||||
self.data["metadata"]["signature"] = signature | |||||
self._cipher = Blowfish.new(sha256(key).digest()) | |||||
print " done." | |||||
else: | |||||
self.data["metadata"]["encryptPasswords"] = False | |||||
try: | |||||
salt = bcrypt.gensalt(self.BCRYPT_ROUNDS) | |||||
signature = bcrypt.hashpw(key, salt) | |||||
self._cipher = Blowfish.new(sha256(key).digest()) | |||||
except ImportError: | |||||
print " error!" | |||||
self._print("""Encryption requires the 'py-bcrypt' and | |||||
'pycrypto' packages:""") | |||||
strt, end = " * \x1b[36m", "\x1b[0m" | |||||
print strt + "http://www.mindrot.org/projects/py-bcrypt/" + end | |||||
print strt + "https://www.dlitz.net/software/pycrypto/" + end | |||||
self._print("""I will disable encryption for now; restart | |||||
configuration after installing these packages if | |||||
you want it.""") | |||||
self._pause() | |||||
else: | |||||
self.data["metadata"]["encryptPasswords"] = True | |||||
self.data["metadata"]["signature"] = signature | |||||
print " done." | |||||
self._print("""The bot can temporarily store its logs in the logs/ | self._print("""The bot can temporarily store its logs in the logs/ | ||||
@@ -95,8 +95,8 @@ class CopyvioMixIn(object): | |||||
if engine == "Yahoo! BOSS": | if engine == "Yahoo! BOSS": | ||||
try: | try: | ||||
oauth.__version__ # Force-load the lazy module | oauth.__version__ # Force-load the lazy module | ||||
except (ImportError, AttributeError): | |||||
e = "The package 'oauth2' could not be imported" | |||||
except ImportError: | |||||
e = "Yahoo! BOSS requires the 'oauth2' package: https://github.com/simplegeo/python-oauth2" | |||||
raise exceptions.UnsupportedSearchEngineError(e) | raise exceptions.UnsupportedSearchEngineError(e) | ||||
return YahooBOSSSearchEngine(credentials) | return YahooBOSSSearchEngine(credentials) | ||||
@@ -530,7 +530,7 @@ class Site(object): | |||||
try: | try: | ||||
self._sql_conn = oursql.connect(**args) | self._sql_conn = oursql.connect(**args) | ||||
except ImportError: | except ImportError: | ||||
e = "Module 'oursql' is required for SQL queries." | |||||
e = "SQL querying requires the 'oursql' package: http://packages.python.org/oursql/" | |||||
raise exceptions.SQLError(e) | raise exceptions.SQLError(e) | ||||
def _get_service_order(self): | def _get_service_order(self): | ||||
@@ -25,24 +25,32 @@ from setuptools import setup, find_packages | |||||
from earwigbot import __version__ | from earwigbot import __version__ | ||||
# Not all of these dependencies are required, particularly the copyvio-specific | |||||
# ones (bs4, lxml, nltk, and oauth2) and the command-specific one (pytz). The | |||||
# bot should run fine without them, but will raise an exception if you try to | |||||
# detect copyvios or run a command that requries one. | |||||
dependencies = [ | |||||
required_deps = [ | |||||
"PyYAML >= 3.10", # Parsing config files | "PyYAML >= 3.10", # Parsing config files | ||||
"beautifulsoup4 >= 4.1.1", # Parsing/scraping HTML for copyvios | |||||
"lxml >= 2.3.5", # Faster parser for BeautifulSoup | |||||
"mwparserfromhell >= 0.1", # Parsing wikicode for manipulation | "mwparserfromhell >= 0.1", # Parsing wikicode for manipulation | ||||
"nltk >= 2.0.2", # Parsing sentences to split article content for copyvios | |||||
"oauth2 >= 1.5.211", # Interfacing with Yahoo! BOSS Search for copyvios | |||||
"oursql >= 0.9.3.1", # Interfacing with MediaWiki databases | |||||
"py-bcrypt >= 0.2", # Hashing the bot key in the config file | |||||
"pycrypto >= 2.6", # Storing bot passwords and keys in the config file | |||||
"pytz >= 2012d", # Handling timezones for the !time IRC command | |||||
] | ] | ||||
extra_deps = { | |||||
"crypto": [ | |||||
"py-bcrypt >= 0.2", # Hashing the bot key in the config file | |||||
"pycrypto >= 2.6", # Storing bot passwords and keys in the config file | |||||
], | |||||
"sql": [ | |||||
"oursql >= 0.9.3.1", # Interfacing with MediaWiki databases | |||||
], | |||||
"copyvios": [ | |||||
"beautifulsoup4 >= 4.1.1", # Parsing/scraping HTML | |||||
"lxml >= 2.3.5", # Faster parser for BeautifulSoup | |||||
"nltk >= 2.0.2", # Parsing sentences to split article content | |||||
"oauth2 >= 1.5.211", # Interfacing with Yahoo! BOSS Search | |||||
], | |||||
"time": [ | |||||
"pytz >= 2012d", # Handling timezones for the !time IRC command | |||||
], | |||||
} | |||||
dependencies = required_deps + sum(extra_deps.values(), []) | |||||
with open("README.rst") as fp: | with open("README.rst") as fp: | ||||
long_docs = fp.read() | long_docs = fp.read() | ||||