]>
gitweb.pimeys.fr Git - bots/saturnin.git/blob - saturnin.py
2 # -*- encoding: utf-8 -*-
6 # Un bot IRC pour remplacer le canard.
7 # parce que le canard, c'est le bien et que braice ne pong pas
12 import socket
, ssl
, json
18 from commands
import getstatusoutput
as ex
20 # Oui, j'ai recodé ma version d'irclib pour pouvoir rattrapper les SIGHUP
21 sys
.path
.insert(0, "/home/vincent/scripts/python-myirclib")
25 # on récupère la config
30 def get_config_logfile(serveur
):
31 serveurs
={"acoeur.crans.org":"acoeur","irc.crans.org":"crans"}
32 return config
.logfile_template
%(serveurs
[serveur
])
34 def log(serveur
,channel
,auteur
=None,message
=None):
35 f
=open(get_config_logfile(serveur
),"a")
36 if auteur
==message
==None:
37 # alors c'est que c'est pas un channel mais juste une ligne de log
38 chain
="%s %s"%(time
.strftime("%F %T"),channel
)
40 chain
="%s [%s:%s] %s"%(time
.strftime("%F %T"),channel
,auteur
,message
)
42 if config
.debug_stdout
:
46 def is_something(chain
,matches
,avant
=u
".*(?:^| )",apres
=u
"(?:$|\.| |,|;).*",case_sensitive
=False,debug
=False):
48 chain
=unicode(chain
,"utf8")
50 chain
=unicode(chain
,"utf8").lower()
51 allmatches
="("+"|".join(matches
)+")"
52 reg
=(avant
+allmatches
+apres
).lower()
56 regexp_pan
= re
.compile(u
".*(" + "|".join(config
.killwords
) + u
").*")
58 return regexp_pan
.match(unicode(chain
,"utf8").lower())
60 class UnicodeBotError(Exception):
62 def bot_unicode(chain
):
65 except UnicodeDecodeError as exc
:
68 class Saturnin(ircbot
.SingleServerIRCBot
):
69 def __init__(self
,serveur
,debug
=False):
70 temporary_pseudo
=config
.irc_pseudo
+str(random
.randrange(10000,100000))
71 ircbot
.SingleServerIRCBot
.__init
__(self
, [(serveur
, 6667)],
72 temporary_pseudo
,"Coin ? ©braice [mais 'faut frapper 20-100]", 10)
75 self
.overops
=config
.overops
76 self
.ops
=self
.overops
+config
.ops
77 self
.chanlist
=config
.chanlist
78 self
.stay_channels
=config
.stay_channels
79 self
.quiet_channels
=config
.quiet_channels
80 self
.play_channels
=config
.play_channels
81 self
.status
= { chan
: [0, None] for chan
in self
.play_channels
}
82 # 0 : pas de spawn prévu
84 # avec en deuxième paramètre le timestamp du moment où il a été déclenché (pas du moment où il se fera)
88 def give_me_my_pseudo(self
,serv
):
89 serv
.privmsg("NickServ","RECOVER %s %s"%(config
.irc_pseudo
,config
.irc_password
))
90 serv
.privmsg("NickServ","RELEASE %s %s"%(config
.irc_pseudo
,config
.irc_password
))
92 serv
.nick(config
.irc_pseudo
)
94 def on_welcome(self
, serv
, ev
):
95 self
.serv
=serv
# ça serv ira :)
96 self
.give_me_my_pseudo(serv
)
97 serv
.privmsg("NickServ","IDENTIFY %s"%(config
.irc_password
))
98 log(self
.serveur
,"Connected")
100 self
.chanlist
= self
.play_channels
= ["#bot"]
101 self
.status
= { chan
: [0, 0] for chan
in self
.play_channels
}
102 for c
in self
.chanlist
:
103 log(self
.serveur
,"JOIN %s"%(c))
105 if c
in self
.play_channels
:
106 spawn_delay
= random
.randrange(*config
.spawn_delays
)
107 self
.spawn(c
, time
.time(), spawn_delay
)
109 def pourmoi(self
, serv
, message
):
110 """renvoie (False,lemessage) ou (True, le message amputé de "pseudo: ")"""
113 if message
[:size
]==pseudo
and len(message
)>size
and message
[size
]==":":
114 return (True,message
[size
+1:].lstrip(" "))
116 return (False,message
)
118 def on_privmsg(self
, serv
, ev
):
119 message
=ev
.arguments()[0]
120 auteur
= irclib
.nm_to_n(ev
.source())
122 test
=bot_unicode(message
)
123 except UnicodeBotError
:
124 if config
.utf8_trigger
:
125 serv
.privmsg(auteur
, random
.choice(config
.utf8_fail_answers
).encode("utf8"))
127 message
=message
.split()
128 cmd
=message
[0].lower()
131 helpdico
={"help":["""HELP <commande>
132 Affiche de l'aide sur la commande""",None,None],
134 Affiche votre score""", None, None],
136 Afficher tous les scores""", None, None],
137 "join": [None, """JOIN <channel>
138 Me fait rejoindre le channel""",None],
139 "leave": [None,"""LEAVE <channel>
140 Me fait quitter le channel (sauf s'il est dans ma stay_list).""",None],
141 "quiet": [None,"""QUIET <channel>
142 Me rend silencieux sur le channel.""",None],
143 "noquiet": [None,"""NOQUIET <channel>
144 Me rend la parole sur le channel.""",None],
145 "play": [None, """PLAY
146 Passe un channel en mode "jouer" """,None],
147 "noplay": [None, """NOPLAY
148 Passe un channel en mode "ne pas jouer" """,None],
149 "SPAWN": [None, """SPAWN <channel>
150 Me fait spawner sur le channel.""",None],
151 "reload": [None,"""RELOAD
152 Recharge la configuration.""",None],
153 "say": [None,None,"""SAY <channel> <message>
154 Me fait parler sur le channel."""],
155 "do": [None,None,"""DO <channel> <action>
156 Me fait faitre une action (/me) sur le channel."""],
157 "stay": [None,None,"""STAY <channel>
158 Ajoute le channel à ma stay_list."""],
159 "nostay": [None,None,"""NOSTAY <channel>
160 Retire le channel de ma stay_list."""],
161 "ops": [None,None,"""OPS
162 Affiche la liste des ops."""],
163 "overops": [None,None,"""OVEROPS
164 Affiche la liste des overops."""],
165 "kick": [None,None,"""KICK <channel> <pseudo> [<raison>]
166 Kicke <pseudo> du channel (Il faut bien entendu que j'y sois op)."""],
167 "die": [None,None,"""DIE
168 Me déconnecte du serveur IRC."""]
170 helpmsg_default
="Liste des commandes disponibles :\nHELP SCORE SCORES"
171 helpmsg_ops
=" JOIN LEAVE QUIET NOQUIET PLAY NOPLAY SPAWN"
172 helpmsg_overops
=" SAY DO STAY NOSTAY OPS OVEROPS KICK DIE"
173 op
,overop
=auteur
in self
.ops
, auteur
in self
.overops
175 helpmsg
=helpmsg_default
179 helpmsg
+=helpmsg_overops
181 helpmsgs
=helpdico
.get(message
[1].lower(),["Commande inconnue.",None,None])
183 if op
and helpmsgs
[1]:
185 helpmsg
+="\n"+helpmsgs
[1]
188 if overop
and helpmsgs
[2]:
190 helpmsg
+="\n"+helpmsgs
[2]
193 for ligne
in helpmsg
.split("\n"):
194 serv
.privmsg(auteur
,ligne
)
196 if auteur
in self
.ops
:
198 if message
[1] in self
.chanlist
:
199 serv
.privmsg(auteur
,"Je suis déjà sur %s"%(message
[1]))
201 serv
.join(message
[1])
202 self
.chanlist
.append(message
[1])
203 serv
.privmsg(auteur
,"Channels : "+" ".join(self
.chanlist
))
204 log(self
.serveur
,"priv",auteur
," ".join(message
))
206 serv
.privmsg(auteur
,"Channels : "+" ".join(self
.chanlist
))
210 if auteur
in self
.ops
and len(message
)>1:
211 if message
[1] in self
.chanlist
:
212 if not (message
[1] in self
.stay_channels
) or auteur
in self
.overops
:
213 self
.quitter(message
[1]," ".join(message
[2:]))
214 self
.chanlist
.remove(message
[1])
215 log(self
.serveur
,"priv",auteur
," ".join(message
)+"[successful]")
217 serv
.privmsg(auteur
,"Non, je reste !")
218 log(self
.serveur
,"priv",auteur
," ".join(message
)+"[failed]")
220 serv
.privmsg(auteur
,"Je ne suis pas sur %s"%(message
[1]))
224 if auteur
in self
.overops
:
226 if message
[1] in self
.stay_channels
:
227 log(self
.serveur
,"priv",auteur
," ".join(message
)+"[failed]")
228 serv
.privmsg(auteur
,"Je stay déjà sur %s."%(message
[1]))
230 log(self
.serveur
,"priv",auteur
," ".join(message
)+"[successful]")
231 self
.stay_channels
.append(message
[1])
232 serv
.privmsg(auteur
,"Stay channels : "+" ".join(self
.stay_channels
))
234 serv
.privmsg(auteur
,"Stay channels : "+" ".join(self
.stay_channels
))
238 if auteur
in self
.overops
:
240 if message
[1] in self
.stay_channels
:
241 log(self
.serveur
,"priv",auteur
," ".join(message
)+"[successful]")
242 self
.stay_channels
.remove(message
[1])
243 serv
.privmsg(auteur
,"Stay channels : "+" ".join(self
.stay_channels
))
245 log(self
.serveur
,"priv",auteur
," ".join(message
)+"[failed]")
246 serv
.privmsg(auteur
,"Je ne stay pas sur %s."%(message
[1]))
251 if auteur
in self
.overops
:
252 log(self
.serveur
,"priv",auteur
," ".join(message
)+"[successful]")
257 if auteur
in self
.ops
:
259 if message
[1] in self
.quiet_channels
:
260 serv
.privmsg(auteur
,"Je me la ferme déjà sur %s"%(message
[1]))
261 log(self
.serveur
,"priv",auteur
," ".join(message
)+"[failed]")
263 self
.quiet_channels
.append(message
[1])
264 serv
.privmsg(auteur
,"Quiet channels : "+" ".join(self
.quiet_channels
))
265 log(self
.serveur
,"priv",auteur
," ".join(message
)+"[successful]")
267 serv
.privmsg(auteur
,"Quiet channels : "+" ".join(self
.quiet_channels
))
271 if auteur
in self
.ops
:
273 if message
[1] in self
.quiet_channels
:
274 self
.quiet_channels
.remove(message
[1])
275 serv
.privmsg(auteur
,"Quiet channels : "+" ".join(self
.quiet_channels
))
276 log(self
.serveur
,"priv",auteur
," ".join(message
)+"[successful]")
278 serv
.privmsg(auteur
,"Je ne me la ferme pas sur %s."%(message
[1]))
279 log(self
.serveur
,"priv",auteur
," ".join(message
)+"[failed]")
283 if auteur
in self
.ops
:
285 if message
[1] in self
.play_channels
:
286 serv
.privmsg(auteur
,"Je play déjà sur %s."%(message
[1]))
287 log(self
.serveur
,"priv",auteur
," ".join(message
)+"[failed]")
289 self
.play_channels
.append(message
[1])
290 self
.spawn(message
[1], 1)
291 serv
.privmsg(auteur
,"Play channels : "+" ".join(self
.play_channels
))
292 log(self
.serveur
,"priv",auteur
," ".join(message
)+"[successful]")
294 serv
.privmsg(auteur
,"Play channels : "+" ".join(self
.play_channels
))
298 if auteur
in self
.ops
:
300 if message
[1] in self
.play_channels
:
301 self
.play_channels
.remove(message
[1])
302 serv
.privmsg(auteur
,"Play channels : "+" ".join(self
.play_channels
))
303 log(self
.serveur
,"priv",auteur
," ".join(message
)+"[successful]")
305 serv
.privmsg(auteur
,"Je ne play pas sur %s."%(message
[1]))
306 log(self
.serveur
,"priv",auteur
," ".join(message
)+"[failed]")
310 if auteur
in self
.ops
:
311 serv
.privmsg(auteur
,"Status : %s" % (self
.status
, ))
315 if auteur
in self
.ops
:
317 if message
[1] in self
.play_channels
:
318 # Le plus pratique pour pas s'embêter c'est de mettre un
319 # delay d'une seconde, comme ça .spawn() fait le boulot
320 self
.spawn(message
[1], time
.time(), 1)
322 serv
.privmsg(auteur
, "Je ne joue pas sur %s" % message
[1])
324 serv
.privmsg(auteur
, "Syntaxe : SPAWN <channel>")
328 if auteur
in self
.overops
and len(message
)>2:
329 serv
.privmsg(message
[1]," ".join(message
[2:]))
330 log(self
.serveur
,"priv",auteur
," ".join(message
))
331 elif len(message
)<=2:
332 serv
.privmsg(auteur
,"Syntaxe : SAY <channel> <message>")
336 if auteur
in self
.overops
and len(message
)>2:
337 serv
.action(message
[1]," ".join(message
[2:]))
338 log(self
.serveur
,"priv",auteur
," ".join(message
))
339 elif len(message
)<=2:
340 serv
.privmsg(auteur
,"Syntaxe : DO <channel> <action>")
344 if auteur
in self
.overops
and len(message
)>2:
345 serv
.kick(message
[1],message
[2]," ".join(message
[3:]))
346 log(self
.serveur
,"priv",auteur
," ".join(message
))
347 elif len(message
)<=2:
348 serv
.privmsg(auteur
,"Syntaxe : KICK <channel> <pseudo> [<raison>]")
352 if auteur
in self
.overops
:
353 serv
.privmsg(auteur
," ".join(self
.ops
))
357 if auteur
in self
.overops
:
358 serv
.privmsg(auteur
," ".join(self
.overops
))
362 if auteur
in self
.ops
:
364 log(self
.serveur
,"priv",auteur
," ".join(message
)+"[successful]")
369 if len(message
) in [3,4] and message
[1].lower()=="transfert":
370 scores
=self
.get_scores()
371 de
,to
=auteur
,message
[2]
372 value
=scores
.get(de
,0)
375 asked
=int(message
[3])
377 serv
.privmsg(auteur
,"Syntaxe : SCORE TRANSFERT <pseudo> [<n>]")
382 serv
.privmsg(auteur
,"Vous n'avez pas de points")
385 serv
.privmsg(auteur
,"Bien tenté…")
388 serv
.privmsg(auteur
,"Vous n'avez que %s points"%(value))
391 self
.add_score(de
,-asked
)
392 self
.add_score(to
,asked
)
393 serv
.privmsg(auteur
,"Transfert de %s points de %s à %s"%(asked
,de
,to
))
395 serv
.privmsg(auteur
,"Syntaxe : SCORE TRANSFERT <pseudo> [<n>]")
397 self
.sendscore(auteur
)
400 self
.sendscores(auteur
)
401 elif auteur
in self
.overops
:
402 souscmd
=message
[1].lower()
406 scores
=self
.get_scores()
407 if scores
.has_key(todelete
):
409 self
.save_scores(scores
)
410 serv
.privmsg(auteur
,"Score de %s supprimé"%(todelete))
412 serv
.privmsg(auteur
,"Ce score n'existe pas : %s"%(todelete))
414 serv
.privmsg(auteur
,"Syntaxe : SCORES DEL <pseudo>")
415 elif souscmd
in ["add","sub"]:
417 toadd
,val
=message
[2],message
[3]
421 serv
.privmsg(auteur
,"Syntaxe : SCORES {ADD|SUB} <pseudo> <n>")
425 self
.add_score(toadd
,val
)
426 serv
.privmsg(auteur
,"Done")
428 serv
.privmsg(auteur
,"Syntaxe : SCORES {ADD|SUB} <pseudo> <n>")
430 serv
.privmsg(auteur
,"Syntaxe : SCORES {DEL|ADD|SUB} <pseudo> [<n>]")
436 serv
.privmsg(auteur
,"Je n'ai pas compris. Essayez HELP…")
438 def sendscore(self
, to
):
439 self
.serv
.privmsg(to
, "Votre score : %s"%(self
.get_scores().get(to
,0)) )
441 def sendscores(self
, to
):
442 scores
=self
.get_scores().items()
444 scores
.sort(lambda x
,y
:cmp(x
[1],y
[1]))
446 self
.serv
.privmsg(to
, "Scores by score : "+" ; ".join(["%s %s"%(i
[0],i
[1]) for i
in scores
]))
448 scores
.sort(lambda x
,y
:cmp(x
[0].lower(),y
[0].lower()))
449 self
.serv
.privmsg(to
, "Scores by pseudo : "+" ; ".join(["%s %s"%(i
[0],i
[1]) for i
in scores
]))
451 def on_pubmsg(self
, serv
, ev
):
452 auteur
= irclib
.nm_to_n(ev
.source())
453 if re
.match(config
.ignored_pseudos
, auteur
):
455 channel
= ev
.target()
456 message
= ev
.arguments()[0]
458 test
=bot_unicode(message
)
459 except UnicodeBotError
:
460 if config
.utf8_trigger
and not channel
in self
.quiet_channels
:
461 serv
.privmsg(channel
, (u
"%s: %s"%(auteur
,random
.choice(config
.utf8_fail_answers
))).encode("utf8"))
463 pour_moi
,message
=self
.pourmoi(serv
,message
)
464 if pour_moi
and message
.split()!=[]:
465 cmd
=message
.split()[0].lower()
467 args
=" ".join(message
.split()[1:])
470 if cmd
in ["meurs","die","crève"]:
471 if auteur
in self
.overops
:
472 log(self
.serveur
,channel
,auteur
,message
+"[successful]")
475 serv
.privmsg(channel
,("%s: %s"%(auteur
,random
.choice(config
.quit_fail_messages
))).encode("utf8"))
476 log(self
.serveur
,channel
,auteur
,message
+"[failed]")
477 elif cmd
== "reload":
478 if auteur
in self
.ops
:
479 log(self
.serveur
, channel
, auteur
, message
+"[successful]")
481 elif cmd
in ["part","leave","dégage","va-t-en","tut'tiresailleurs,c'estmesgalets"]:
482 if auteur
in self
.ops
and (not (channel
in self
.stay_channels
)
483 or auteur
in self
.overops
):
484 self
.quitter(channel
)
485 log(self
.serveur
,channel
,auteur
,message
+"[successful]")
486 if channel
in self
.chanlist
:
487 self
.chanlist
.remove(channel
)
489 serv
.privmsg(channel
,("%s: %s"%(auteur
,random
.choice(config
.leave_fail_messages
))).encode("utf8"))
490 log(self
.serveur
,channel
,auteur
,message
+"[failed]")
492 self
.sendscore(auteur
)
493 elif cmd
== "scores":
494 self
.sendscores(auteur
)
497 self
.shot(channel
, auteur
)
499 def on_action(self
, serv
, ev
):
500 action
= ev
.arguments()[0]
501 auteur
= irclib
.nm_to_n(ev
.source())
502 channel
= ev
.target()
504 #~ test=bot_unicode(action)
505 #~ except UnicodeBotError:
506 #~ if config.utf8_trigger and not channel in self.quiet_channels:
507 #~ serv.privmsg(channel, (u"%s: %s"%(auteur,random.choice(config.utf8_fail_answers))).encode("utf8"))
509 #~ mypseudo=self.nick
511 def on_kick(self
,serv
,ev
):
512 auteur
= irclib
.nm_to_n(ev
.source())
513 channel
= ev
.target()
514 victime
= ev
.arguments()[0]
515 raison
= ev
.arguments()[1]
516 if victime
==self
.nick
:
517 log(self
.serveur
,"%s kické de %s par %s (raison : %s)" %(victime
,channel
,auteur
,raison
))
520 #~ l1,l2=config.kick_answers,config.kick_actions
521 #~ n1,n2=len(l1),len(l2)
522 #~ i=random.randrange(n1+n2)
524 #~ serv.action(channel,l2[i-n1].format(auteur).encode("utf8"))
526 #~ serv.privmsg(channel,l1[i].format(auteur).encode("utf8"))
528 def spawn(self
, channel
, timestamp
, delay
=0):
529 if channel
in self
.play_channels
:
531 self
.serv
.execute_delayed(delay
, self
.spawn
, (channel
, timestamp
))
532 self
.status
[channel
] = [1, timestamp
]
534 # on teste le timestamp pour pas s'emmêler dans les spawn
535 infos
= self
.status
.get(channel
, [0,0])
536 if infos
== [1, timestamp
]:
537 spawn_sentence
= random
.choice(config
.canards
) + random
.choice(config
.spawn_sentences
)
538 self
.serv
.privmsg(channel
, spawn_sentence
.encode("utf8"))
539 self
.status
[channel
] = [2, timestamp
]
540 times_up_delay
= random
.randrange(*config
.times_up_delays
)
541 self
.serv
.execute_delayed(times_up_delay
, self
.too_slow
, (channel
, timestamp
))
543 def too_slow(self
, channel
, timestamp
):
544 infos
= self
.status
.get(channel
, [0,0])
545 if infos
== [2, timestamp
]:
546 self
.serv
.privmsg(channel
, random
.choice(config
.times_up_sentences
).encode("utf8"))
547 respawn_delay
= random
.randrange(*config
.spawn_delays
)
548 self
.spawn(channel
, time
.time(), respawn_delay
)
550 def shot(self
, channel
, auteur
):
551 if self
.status
.get(channel
, [0, 0])[0] == 2:
552 succeed
= random
.randrange(0,101) > config
.proba_miss
554 self
.serv
.privmsg(channel
, random
.choice(config
.killed_templates
).format(auteur
).encode("utf8"))
555 self
.add_score(auteur
, 1)
556 if random
.randrange(0, 101) < config
.proba_killed_sentence
:
557 self
.serv
.privmsg(channel
, random
.choice(config
.killed_sentences
).encode("utf8"))
558 respawn_delay
= random
.randrange(*config
.spawn_delays
)
559 self
.spawn(channel
, time
.time(), respawn_delay
)
561 self
.serv
.privmsg(channel
, random
.choice(config
.miss_templates
).format(auteur
).encode("utf8"))
562 if random
.randrange(0,101) < config
.proba_miss_sentence
:
563 self
.serv
.privmsg(channel
, random
.choice(config
.miss_sentences
).encode("utf8"))
565 def quitter(self
,chan
,leave_message
=None):
566 if leave_message
==None:
567 leave_message
=random
.choice(config
.leave_messages
)
568 self
.serv
.part(chan
,message
=leave_message
.encode("utf8"))
571 quit_message
=random
.choice(config
.quit_messages
)
572 self
.die(msg
=quit_message
.encode("utf8"))
574 def get_scores(self
):
575 f
=open(config
.score_file
)
576 scores
=pickle
.load(f
)
580 def add_score(self
, pseudo
, value
):
581 scores
=self
.get_scores()
582 if scores
.has_key(pseudo
):
583 scores
[pseudo
]+=value
586 self
.save_scores(scores
)
588 def save_scores(self
,scores
):
589 f
=open(config
.score_file
,"w")
590 pickle
.dump(scores
,f
)
594 return self
.serv
.get_nickname()
595 nick
=property(_getnick
)
597 def reload(self
, auteur
=None):
599 if auteur
in [None, "SIGHUP"]:
600 towrite
= "Config reloaded" + " (SIGHUP received)"*(auteur
== "SIGHUP")
601 for to
in config
.report_bugs_to
:
602 self
.serv
.privmsg(to
, towrite
)
603 log(self
.serveur
, towrite
)
605 self
.serv
.privmsg(auteur
,"Config reloaded")
607 def start_as_daemon(self
, outfile
):
608 sys
.stderr
= Logger(outfile
)
612 class Logger(object):
613 """Pour écrire ailleurs que sur stdout"""
614 def __init__(self
, filename
="saturnin.full.log"):
615 self
.filename
= filename
617 def write(self
, message
):
618 f
= open(self
.filename
, "a")
623 if __name__
=="__main__":
626 print "Usage : saturnin.py <serveur> [--debug] [--no-output] [--daemon [--pidfile]] [--outfile]"
627 print " --outfile sans --no-output ni --daemon n'a aucun effet"
630 if "--daemon" in sys
.argv
:
631 thisfile
= os
.path
.realpath(__file__
)
632 thisdirectory
= thisfile
.rsplit("/", 1)[0]
633 os
.chdir(thisdirectory
)
637 if "debug" in sys
.argv
or "--debug" in sys
.argv
:
641 if "--no-output" in sys
.argv
or "--daemon" in sys
.argv
:
642 outfile
= "/var/log/bots/saturnin.full.log"
645 if arg
[0].strip('-') in ["out", "outfile", "logfile"]:
647 sys
.stdout
= Logger(outfile
)
648 if "--quiet" in sys
.argv
:
649 config
.debug_stdout
=False
650 serveurs
={"a♡":"acoeur.crans.org","acoeur":"acoeur.crans.org","acoeur.crans.org":"acoeur.crans.org",
651 "irc":"irc.crans.org","crans":"irc.crans.org","irc.crans.org":"irc.crans.org",
654 serveur
=serveurs
[serveur
]
656 print "Server Unknown : %s"%(serveur)
658 saturnin
= Saturnin(serveur
,debug
)
659 # Si on reçoit un SIGHUP, on reload la config
660 def sighup_handler(signum
, frame
):
661 saturnin
.reload("SIGHUP")
662 signal
.signal(signal
.SIGHUP
, sighup_handler
)
664 child_pid
= os
.fork()
667 saturnin
.start_as_daemon(outfile
)
669 # on enregistre le pid de saturnin
670 pidfile
= "/var/run/bots/saturnin.pid"
673 if arg
[0].strip('-') in ["pidfile"]:
675 f
= open(pidfile
, "w")
676 f
.write("%s\n" % child_pid
)