2003-08-28 18:33:45 +02:00
|
|
|
#!/usr/bin/env python
|
|
|
|
|
|
|
|
###
|
2003-10-12 17:12:18 +02:00
|
|
|
# Copyright (c) 2003, Jeremiah Fincher
|
2003-08-28 18:33:45 +02:00
|
|
|
# All rights reserved.
|
|
|
|
#
|
|
|
|
# Redistribution and use in source and binary forms, with or without
|
|
|
|
# modification, are permitted provided that the following conditions are met:
|
|
|
|
#
|
|
|
|
# * Redistributions of source code must retain the above copyright notice,
|
|
|
|
# this list of conditions, and the following disclaimer.
|
|
|
|
# * Redistributions in binary form must reproduce the above copyright notice,
|
|
|
|
# this list of conditions, and the following disclaimer in the
|
|
|
|
# documentation and/or other materials provided with the distribution.
|
|
|
|
# * Neither the name of the author of this software nor the name of
|
|
|
|
# contributors to this software may be used to endorse or promote products
|
|
|
|
# derived from this software without specific prior written consent.
|
|
|
|
#
|
|
|
|
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
|
|
|
# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
|
|
# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
|
|
# ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
|
|
|
|
# LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
|
|
|
# CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
|
|
|
# SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
|
|
# INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
|
|
|
# CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
|
|
# ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
|
|
# POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
###
|
|
|
|
|
|
|
|
"""
|
2003-10-23 09:32:19 +02:00
|
|
|
This is the template for bots. supybot-wizard uses this file to make
|
2003-08-28 18:33:45 +02:00
|
|
|
customized startup files for bots.
|
|
|
|
"""
|
|
|
|
|
|
|
|
import re
|
2003-09-29 08:12:41 +02:00
|
|
|
import os
|
2003-08-28 18:33:45 +02:00
|
|
|
import sys
|
2003-09-09 10:04:25 +02:00
|
|
|
|
|
|
|
if sys.version_info < (2, 3, 0):
|
|
|
|
sys.stderr.write('This program requires Python >= 2.3.0\n')
|
|
|
|
sys.exit(-1)
|
|
|
|
|
2003-09-29 08:12:41 +02:00
|
|
|
if os.name == 'posix':
|
|
|
|
if os.getuid() == 0 or os.geteuid() == 0:
|
|
|
|
sys.stderr.write('Dude, don\'t even try to run this as root.\n')
|
|
|
|
sys.exit(-1)
|
|
|
|
|
2003-08-28 18:33:45 +02:00
|
|
|
import time
|
|
|
|
import optparse
|
|
|
|
|
|
|
|
started = time.time()
|
2003-08-30 21:20:08 +02:00
|
|
|
|
2003-09-24 09:27:13 +02:00
|
|
|
import supybot
|
2003-08-28 18:33:45 +02:00
|
|
|
|
|
|
|
import conf
|
|
|
|
|
2003-09-24 09:27:13 +02:00
|
|
|
defaultNick = "%%nick%%"
|
|
|
|
defaultUser = "%%user%%"
|
|
|
|
defaultIdent = "%%ident%%"
|
|
|
|
defaultServer = "%%server%%"
|
|
|
|
defaultPassword = "%%password%%"
|
2003-09-15 19:08:24 +02:00
|
|
|
|
2003-09-24 09:27:13 +02:00
|
|
|
conf.commandsOnStart = "%%onStart%%"
|
2003-08-28 18:33:45 +02:00
|
|
|
|
2003-09-24 09:27:13 +02:00
|
|
|
afterConnect = "%%afterConnect%%"
|
2003-08-28 18:33:45 +02:00
|
|
|
|
2003-09-25 09:15:39 +02:00
|
|
|
debugVariables = "%%debugVariables%%"
|
2003-09-24 09:27:13 +02:00
|
|
|
configVariables = "%%configVariables%%"
|
2003-08-28 18:33:45 +02:00
|
|
|
|
2003-09-24 09:27:13 +02:00
|
|
|
if not isinstance(configVariables, basestring):
|
|
|
|
for (name, value) in configVariables.iteritems():
|
|
|
|
setattr(conf, name, value)
|
2003-08-28 18:33:45 +02:00
|
|
|
|
2003-10-20 09:18:43 +02:00
|
|
|
if not os.path.exists(conf.dataDir):
|
|
|
|
os.mkdir(conf.dataDir)
|
|
|
|
if not os.path.exists(conf.logDir):
|
|
|
|
os.mkdir(conf.logDir)
|
|
|
|
if not os.path.exists(conf.confDir):
|
|
|
|
os.mkdir(conf.confDir)
|
|
|
|
|
2003-10-12 14:43:24 +02:00
|
|
|
import debug
|
2003-08-28 18:33:45 +02:00
|
|
|
|
2003-10-12 14:43:24 +02:00
|
|
|
if not isinstance(debugVariables, basestring):
|
|
|
|
for (name, value) in debugVariables.iteritems():
|
|
|
|
setattr(debug, name, value)
|
2003-09-25 09:15:39 +02:00
|
|
|
|
2003-10-12 14:43:24 +02:00
|
|
|
def main():
|
2003-08-28 18:33:45 +02:00
|
|
|
import world
|
|
|
|
import drivers
|
2003-09-01 09:15:18 +02:00
|
|
|
import schedule
|
|
|
|
schedule.addPeriodicEvent(world.upkeep, 300)
|
2003-08-28 18:33:45 +02:00
|
|
|
world.startedAt = started
|
|
|
|
try:
|
|
|
|
while world.ircs:
|
|
|
|
drivers.run()
|
|
|
|
except:
|
|
|
|
try:
|
|
|
|
debug.recoverableException()
|
|
|
|
except: # It must've been deadly for a reason :)
|
|
|
|
sys.exit(0)
|
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
###
|
|
|
|
# Options:
|
|
|
|
# -p (profiling)
|
|
|
|
# -O (optimizing)
|
|
|
|
# -n, --nick (nick)
|
|
|
|
# -s, --server (server)
|
|
|
|
# --startup (commands to run onStart)
|
|
|
|
# --connect (commands to run afterConnect)
|
|
|
|
# --config (configuration values)
|
|
|
|
parser = optparse.OptionParser(usage='Usage: %prog [options]',
|
|
|
|
version='supybot %s' % conf.version)
|
|
|
|
parser.add_option('-P', '--profile', action='store_true', dest='profile',
|
|
|
|
help='enables profiling')
|
|
|
|
parser.add_option('-O', action='count', dest='optimize',
|
|
|
|
help='-O optimizes asserts out of the code; ' \
|
|
|
|
'-OO optimizes asserts and uses psyco.')
|
|
|
|
parser.add_option('-n', '--nick', action='store',
|
2003-09-15 19:08:24 +02:00
|
|
|
dest='nick', default=defaultNick,
|
2003-09-25 09:15:39 +02:00
|
|
|
help='nick the bot should use')
|
2003-08-28 18:33:45 +02:00
|
|
|
parser.add_option('-s', '--server', action='store',
|
2003-09-15 19:08:24 +02:00
|
|
|
dest='server', default=defaultServer,
|
2003-09-25 09:15:39 +02:00
|
|
|
help='server to connect to')
|
2003-08-28 18:33:45 +02:00
|
|
|
parser.add_option('-u', '--user', action='store',
|
2003-09-15 19:08:24 +02:00
|
|
|
dest='user', default=defaultUser,
|
2003-08-28 18:33:45 +02:00
|
|
|
help='full username the bot should use')
|
|
|
|
parser.add_option('-i', '--ident', action='store',
|
2003-09-15 19:08:24 +02:00
|
|
|
dest='ident', default=defaultIdent,
|
2003-08-28 18:33:45 +02:00
|
|
|
help='ident the bot should use')
|
|
|
|
parser.add_option('-p', '--password', action='store',
|
2003-09-15 19:08:24 +02:00
|
|
|
dest='password', default=defaultPassword,
|
2003-08-28 18:33:45 +02:00
|
|
|
help='server password the bot should use')
|
|
|
|
parser.add_option('--startup', action='append', dest='onStart',
|
|
|
|
help='file of additional commands to run at startup.')
|
|
|
|
parser.add_option('--connect', action='append', dest='afterConnect',
|
|
|
|
help='file of additional commands to run after connect')
|
|
|
|
parser.add_option('--config', action='append', dest='conf',
|
|
|
|
help='file of configuration variables to set')
|
|
|
|
|
|
|
|
(options, args) = parser.parse_args()
|
|
|
|
|
|
|
|
if options.optimize:
|
|
|
|
__builtins__.__debug__ = False
|
|
|
|
if options.optimize > 1:
|
|
|
|
import psyco
|
|
|
|
psyco.full()
|
|
|
|
|
|
|
|
if options.onStart:
|
|
|
|
for filename in options.onStart:
|
|
|
|
fd = file(filename)
|
|
|
|
for line in fd:
|
|
|
|
conf.commandsOnStart.append(line.rstrip())
|
|
|
|
fd.close()
|
|
|
|
|
|
|
|
if options.afterConnect:
|
|
|
|
for filename in options.afterConnect:
|
|
|
|
fd = file(filename)
|
|
|
|
for line in fd:
|
|
|
|
afterConnect.append(line.rstrip())
|
|
|
|
fd.close()
|
|
|
|
|
|
|
|
assignmentRe = re.compile('\s*[:=]\s*')
|
|
|
|
if options.conf:
|
|
|
|
for filename in options.conf:
|
|
|
|
fd = file(filename)
|
|
|
|
for line in fd:
|
|
|
|
(name, valueString) = assignmentRe.split(line.rstrip(), 1)
|
|
|
|
try:
|
|
|
|
value = eval(valueString)
|
|
|
|
except Exception, e:
|
|
|
|
sys.stderr.write('Invalid configuration value: %r' % \
|
|
|
|
valueString)
|
|
|
|
sys.exit(-1)
|
|
|
|
|
|
|
|
nick = options.nick
|
|
|
|
user = options.user
|
|
|
|
ident = options.ident
|
|
|
|
password = options.password
|
|
|
|
|
|
|
|
if ':' in options.server:
|
|
|
|
serverAndPort = options.server.split(':', 1)
|
|
|
|
serverAndPort[1] = int(serverAndPort[1])
|
|
|
|
server = tuple(serverAndPort)
|
|
|
|
else:
|
|
|
|
server = (options.server, 6667)
|
|
|
|
|
|
|
|
import irclib
|
|
|
|
import ircmsgs
|
|
|
|
import drivers
|
|
|
|
import callbacks
|
2003-10-21 08:03:57 +02:00
|
|
|
import Owner
|
2003-08-28 18:33:45 +02:00
|
|
|
|
2003-09-30 15:56:40 +02:00
|
|
|
class ConfigAfterConnect(irclib.IrcCallback):
|
2003-08-28 18:33:45 +02:00
|
|
|
public = False
|
|
|
|
def __init__(self, commands):
|
|
|
|
self.commands = commands
|
|
|
|
def do376(self, irc, msg):
|
|
|
|
for command in self.commands:
|
|
|
|
msg = ircmsgs.privmsg(irc.nick, command, prefix=irc.prefix)
|
|
|
|
irc.queueMsg(msg)
|
|
|
|
do377 = do376
|
2003-09-30 15:56:40 +02:00
|
|
|
do422 = do376 # MOTD File is missing.
|
2003-08-28 18:33:45 +02:00
|
|
|
|
|
|
|
# We pre-tokenize the commands just to save on significant amounts of work.
|
|
|
|
conf.commandsOnStart = map(callbacks.tokenize, conf.commandsOnStart)
|
|
|
|
|
|
|
|
irc = irclib.Irc(nick, user, ident, password)
|
2003-10-21 08:03:57 +02:00
|
|
|
callback = Owner.Class()
|
2003-08-28 18:33:45 +02:00
|
|
|
callback.configure(irc)
|
|
|
|
irc.addCallback(callback)
|
2003-09-30 15:56:40 +02:00
|
|
|
irc.addCallback(ConfigAfterConnect(afterConnect))
|
2003-08-28 18:33:45 +02:00
|
|
|
driver = drivers.newDriver(server, irc)
|
|
|
|
|
|
|
|
if options.profile:
|
|
|
|
import profile
|
2003-10-09 00:33:09 +02:00
|
|
|
profile.run('main()', '%s-%i.prof' % (nick, time.time()))
|
2003-08-28 18:33:45 +02:00
|
|
|
else:
|
|
|
|
main()
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# vim:set shiftwidth=4 tabstop=8 expandtab textwidth=78:
|