This repository has been archived by the owner on May 26, 2018. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 6
/
ethicsbot.py
178 lines (145 loc) · 5.62 KB
/
ethicsbot.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
#!/usr/bin/env python
"""
ethicsbot: Acts as automated ethical consultant on IRC. Not a lawyer. Like a magic 8-ball, except always gives the same answer to the same question.
Copyright 2010 - 2012 Michael Farrell <http://micolous.id.au>
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU Affero General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU Affero General Public License for more details.
You should have received a copy of the GNU Affero General Public License
along with this program. If not, see <http://www.gnu.org/licenses/>.
"""
import asyncore, random
from hashlib import sha512
from datetime import datetime, timedelta
from configparser_plus import ConfigParserPlus
from sys import argv, exit
from ircasync import *
from subprocess import Popen, PIPE
ETHICAL_MESSAGES = [
[ # unethical
'your chairman resigns after the plan is revealed to the public by "60 Minutes".',
'your company was fined 1.4m$ after a probe into your activities.',
'your company was fined a record 600m$ by the EU after reports of bribery.',
'accounting irregularities were exposed by a whistleblower, resulting in a fine for your company.',
'the prime minister announces new taxes on employees because of a percieved lack of local investment by your company in rural areas.',
],
[ # ethical
'a jury found no misconduct by board members.',
'testimony of key witnesses had to be striken from the record after finding they were all under the influence of drugs.',
'the prime minister issued a public apology over the government\'s handling of your company\'s case in the media.',
],
[ # unsure
'a royal commission is establisted into your activities.',
'the government announces a bailout package for your company.',
'your company\'s charity activities in the third world were recognised by the judge in giving your company a lenient sentence.',
'your company owns a 70% share in local newspapers and television stations, meaning this incident is never brought to the public\'s attention.',
],
]
ETHICAL_STATES = [
'unethical',
'ethical',
'unsure',
]
ETHICAL_STATES_COUNT = len(ETHICAL_STATES)
DEFAULT_CONFIG = {
'ethicsbot': {
'server': 'localhost',
'port': DEFAULT_PORT,
'ipv6': 'no',
'nick': 'ethicsbot',
'channel': '#test',
'flood_cooldown': 5,
'version': 'ethicsbot; https://github.com/micolous/ircbots/',
}
}
config = ConfigParserPlus(DEFAULT_CONFIG)
try:
config.readfp(open(argv[1]))
except:
try:
config.readfp(open('ethicsbot.ini'))
except:
print "Syntax:"
print " %s [config]" % argv[0]
print ""
print "If no configuration file is specified or there was an error, it will default to `ethicsbot.ini'."
print "If there was a failure reading the configuration, it will display this message."
exit(1)
# read config
SERVER = config.get('ethicsbot', 'server')
PORT = config.getint('ethicsbot', 'port')
IPV6 = config.getboolean('ethicsbot', 'ipv6')
NICK = config.get('ethicsbot', 'nick')
CHANNEL = config.get('ethicsbot', 'channel')
VERSION = config.get('ethicsbot', 'version') + '; %s'
try: VERSION = VERSION % Popen(["git","branch","-v","--contains"], stdout=PIPE).communicate()[0].strip()
except: VERSION = VERSION % 'unknown'
del Popen, PIPE
FLOOD_COOLDOWN = timedelta(seconds=config.getint('ethicsbot', 'flood_cooldown'))
try: NICKSERV_PASS = config.get('ethicsbot', 'nickserv_pass')
except: NICKSERV_PASS = None
message_buffer = []
last_message = datetime.now()
flooders = []
ignore_list = []
if config.has_section('ignore'):
for k,v in config.items('ignore'):
try:
ignore_list.append(re.compile(v, re.I))
except Exception, ex:
print "Error compiling regular expression in ignore list (%s):" % k
print " %s" % v
print ex
exit(1)
# main code
def handle_msg(event, match):
global message_buffer, MAX_MESSAGES, last_message, flooders, CHANNEL
msg = event.text
if event.channel.lower() != CHANNEL.lower():
# ignore messages not from our channel
return
if msg.startswith('?ethical'):
for item in ignore_list:
if item.search(event.origin) != None:
# ignore list item hit
print "Ignoring message from %s because of: %s" % (event.origin, item.pattern)
return
# now flood protect!
delta = event.when - last_message
last_message = event.when
if delta < FLOOD_COOLDOWN:
# 5 seconds between requests
# any more are ignored
print "Flood protection hit, %s of %s seconds were waited" % (delta.seconds, FLOOD_COOLDOWN.seconds)
return
parts = msg.split(' ')
query = (''.join(parts[1:])).lower()
if len(query) == 0:
event.reply("%s: you must give me an ethical conundrum to process!" % event.nick)
return
# hash the request
h = sha512()
h.update(query)
ethical = 0
for c in h.digest():
for x in xrange(0,8):
if ord(c) & (2 ** x) > 0:
ethical += 1
ethical %= ETHICAL_STATES_COUNT
event.reply('%s: (%s) %s' % (event.nick, ETHICAL_STATES[ethical], random.choice(ETHICAL_MESSAGES[ethical])))
def handle_welcome(event, match):
global NICKSERV_PASS
# Compliance with most network's rules to set this mode on connect.
event.connection.usermode("+B")
if NICKSERV_PASS != None:
event.connection.todo(['NickServ', 'identify', NICKSERV_PASS])
irc = IRC(nick=NICK, start_channels=[CHANNEL], version=VERSION)
irc.bind(handle_msg, PRIVMSG)
irc.bind(handle_welcome, RPL_WELCOME)
irc.make_conn(SERVER, PORT, ipv6=IPV6)
asyncore.loop()