3 # Codé par 20-100 (commencé le 21/06/12)
5 # Un bot IRC pour kicker à tour de bras de #déprime
10 import socket
, ssl
, json
17 # Oui, j'ai recodé ma version d'irclib pour pouvoir rattrapper les SIGHUP
18 sys
.path
.insert(0, "/home/vincent/scripts/python-myirclib")
27 def get_config_logfile(serveur
):
28 serveurs
={"acoeur.crans.org":"acoeur","irc.crans.org":"crans"}
29 return config
.logfile_template
%(serveurs
[serveur
])
32 def log(serveur
,channel
,auteur
=None,message
=None):
33 f
=open(get_config_logfile(serveur
),"a")
34 if auteur
==message
==None:
35 # alors c'est que c'est pas un channel mais juste une ligne de log
36 chain
="%s %s"%(time
.strftime("%F %T"),channel
)
38 chain
="%s [%s:%s] %s"%(time
.strftime("%F %T"),channel
,auteur
,message
)
40 if config
.debug_stdout
:
44 reg_is_smiley
= re
.compile(u
".*("+u
"|".join(config
.smileys
)+u
")")
46 chain
=unicode(chain
,"utf8")
47 o
=re
.match(reg_is_smiley
,chain
)
50 reg_is_anglicisme
= re
.compile(u
".*(?:^| )(" + u
"|".join(config
.anglicismes
) + u
")(?:$|\.| |,|;)")
51 def is_anglicisme(chain
):
52 chain
= unicode(chain
, "utf8").lower()
53 o
= re
.match(reg_is_anglicisme
, chain
)
56 # Cette liste contient la liste des raisons pour lesquelles on peut se faire kicker
57 # chaque élément contient :
58 # - la fonction de détection du kick (qui matchera une regexp et renverra l'objet de match)
59 # - la raison donnée au moment du kick ({0} étant remplacé par ce qui a matché)
60 config
.kicking_list
= [
61 [is_smiley
, u
'"{0}" ? Ici on déprime.'],
62 [is_anglicisme
, u
'"{0}" ? Get out, you and your fucking anglicism !']
65 def is_something(chain
,matches
,avant
=u
".*(?:^| )",apres
=u
"(?:$|\.| |,|;).*",case_sensitive
=False,debug
=False):
67 chain
=unicode(chain
,"utf8")
69 chain
=unicode(chain
,"utf8").lower()
70 allmatches
="("+"|".join(matches
)+")"
71 reg
=(avant
+allmatches
+apres
).lower()
75 def is_insult(chain
,debug
=True):
76 return is_something(chain
,config
.insultes
,avant
=".*(?:^| |')")
77 def is_not_insult(chain
):
78 chain
=unicode(chain
,"utf8")
79 insult_regexp
=u
"("+u
"|".join(config
.insultes
)+u
")"
80 middle_regexp
=u
"(une? (?:(?:putain|enfoiré) d(?:e |'))*|)(?:| super )(?: (?:gros|petit|grand|énorme) |)"
81 reg
=".*pas %s%s.*"%(middle_regexp
,insult_regexp
)
82 if re
.match(reg
,chain
):
86 def is_compliment(chain
,debug
=True):
87 return is_something(chain
,config
.compliment_triggers
,avant
=".*(?:^| |')")
89 return is_something(chain
,config
.perdu
)
91 return is_something(chain
,config
.tag_triggers
)
93 return is_something(chain
,config
.tesla_triggers
,avant
=u
"^",apres
=u
"$",debug
=True)
95 return is_something(chain
,config
.merci_triggers
)
97 return is_something(chain
,config
.tamere_triggers
)
98 def is_bad_action_trigger(chain
,pseudo
):
99 return is_something(chain
,config
.bad_action_triggers
,avant
=u
"^",
100 apres
="(?: [a-z]*ment)? %s($|\.| |,|;).*"%(pseudo))
101 def is_good_action_trigger(chain
,pseudo
):
102 return is_something(chain
,config
.good_action_triggers
,avant
=u
"^",
103 apres
="(?: [a-z]*ment)? %s($|\.| |,|;).*"%(pseudo))
104 def is_bonjour(chain
):
105 return is_something(chain
,config
.bonjour_triggers
,avant
=u
"^")
106 def is_bonne_nuit(chain
):
107 return is_something(chain
,config
.bonne_nuit_triggers
,avant
=u
"^")
109 return re
.match(u
"^(pan|bim|bang)( .*)?$",unicode(chain
,"utf8").lower().strip())
112 _
,_
,_
,h
,m
,s
,_
,_
,_
=time
.localtime()
113 return (conf
[0],0,0)<(h
,m
,s
)<(conf
[1],0,0)
115 return is_time(config
.daytime
)
117 return is_time(config
.nighttime
)
120 class UnicodeBotError(Exception):
122 def bot_unicode(chain
):
124 unicode(chain
,"utf8")
125 except UnicodeDecodeError as exc
:
126 raise UnicodeBotError
128 class Themis(ircbot
.SingleServerIRCBot
):
129 def __init__(self
,serveur
,debug
=False):
130 temporary_pseudo
=config
.irc_pseudo
+str(random
.randrange(10000,100000))
131 ircbot
.SingleServerIRCBot
.__init
__(self
, [(serveur
, 6667)],
132 temporary_pseudo
,"Des[bot]e de #déprime", 10)
135 self
.overops
=config
.overops
136 self
.ops
=self
.overops
+config
.ops
137 self
.report_bugs_to
=config
.report_bugs_to
138 self
.chanlist
=config
.chanlist
139 self
.kick_channels
=config
.kick_channels
140 self
.stay_channels
=config
.stay_channels
141 self
.quiet_channels
=config
.quiet_channels
145 def give_me_my_pseudo(self
,serv
):
146 serv
.privmsg("NickServ","RECOVER %s %s"%(config
.irc_pseudo
,config
.irc_password
))
147 serv
.privmsg("NickServ","RELEASE %s %s"%(config
.irc_pseudo
,config
.irc_password
))
149 serv
.nick(config
.irc_pseudo
)
151 def give_me_my_op_status(self
,serv
,chan
):
152 serv
.privmsg("ChanServ","OP %s"%(chan))
154 def on_welcome(self
, serv
, ev
):
155 self
.serv
=serv
# ça serv ira
156 self
.give_me_my_pseudo(serv
)
157 serv
.privmsg("NickServ","identify %s"%(config
.irc_password
))
158 log(self
.serveur
,"Connected")
160 self
.chanlist
=["#bot"]
161 self
.kick_channels
=["#bot"]
162 for c
in self
.chanlist
:
163 log(self
.serveur
,"JOIN %s"%(c))
165 self
.give_me_my_op_status(serv
,c
)
168 def lost(self
,serv
,channel
,forced
=False):
169 if self
.last_perdu
+config
.time_between_perdu
<time
.time() or forced
:
170 if not channel
in self
.quiet_channels
or forced
:
171 serv
.privmsg(channel
,"J'ai perdu !")
172 self
.last_perdu
=time
.time()
173 delay
=config
.time_between_perdu_trigger
174 delta
=config
.time_between_perdu_trigger_delta
175 serv
.execute_delayed(random
.randrange(delay
-delta
,delay
+delta
),self
.lost
,(serv
,channel
))
177 def pourmoi(self
, serv
, message
):
178 """renvoie (False,lemessage) ou (True, le message amputé de "pseudo: ")"""
181 if message
[:size
]==pseudo
and len(message
)>size
and message
[size
]==":":
182 return (True,message
[size
+1:].lstrip(" "))
184 return (False,message
)
186 def on_privmsg(self
, serv
, ev
):
187 message
=ev
.arguments()[0]
188 auteur
= irclib
.nm_to_n(ev
.source())
190 test
=bot_unicode(message
)
191 except UnicodeBotError
:
192 if config
.utf8_trigger
:
193 serv
.privmsg(auteur
, config
.utf8_fail
)
195 message
=message
.split()
196 cmd
=message
[0].lower()
199 helpdico
={"help":["""HELP <commande>
200 Affiche de l'aide sur la commande""",None,None],
201 "join": [None, """JOIN <channel>
202 Me fait rejoindre le channel""",None],
203 "leave": [None,"""LEAVE <channel>
204 Me fait quitter le channel (sauf s'il est dans ma stay_list).""",None],
205 "quiet": [None,"""QUIET <channel>
206 Me rend silencieux sur le channel.""",None],
207 "noquiet": [None,"""NOQUIET <channel>
208 Me rend la parole sur le channel.""",None],
209 "reload": [None,"""RELOAD
210 Recharge la configuration.""",None],
211 "say": [None,None,"""SAY <channel> <message>
212 Me fait parler sur le channel."""],
213 "do": [None,None,"""DO <channel> <action>
214 Me fait faitre une action (/me) sur le channel."""],
215 "stay": [None,None,"""STAY <channel>
216 Ajoute le channel à ma stay_list."""],
217 "nostay": [None,None,"""NOSTAY <channel>
218 Retire le channel de ma stay_list."""],
219 "ops": [None,None,"""OPS
220 Affiche la liste des ops."""],
221 "overops": [None,None,"""OVEROPS
222 Affiche la liste des overops."""],
223 "kick": [None,None,"""KICK <channel> <pseudo> [<raison>]
224 Kicke <pseudo> du channel (Il faut bien entendu que j'y sois op)."""],
225 "die": [None,None,"""DIE
226 Me déconnecte du serveur IRC."""]
228 helpmsg_default
="Liste des commandes disponibles :\nHELP"
229 helpmsg_ops
=" JOIN LEAVE QUIET NOQUIET LOST RELOAD"
230 helpmsg_overops
=" SAY DO STAY NOSTAY OPS OVEROPS KICK DIE"
231 op
,overop
=auteur
in self
.ops
, auteur
in self
.overops
233 helpmsg
=helpmsg_default
237 helpmsg
+=helpmsg_overops
239 helpmsgs
=helpdico
.get(message
[1].lower(),["Commande inconnue.",None,None])
241 if op
and helpmsgs
[1]:
243 helpmsg
+="\n"+helpmsgs
[1]
246 if overop
and helpmsgs
[2]:
248 helpmsg
+="\n"+helpmsgs
[2]
251 for ligne
in helpmsg
.split("\n"):
252 serv
.privmsg(auteur
,ligne
)
254 if auteur
in self
.ops
:
256 if message
[1] in self
.chanlist
:
257 serv
.privmsg(auteur
,"Je suis déjà sur %s"%(message
[1]))
259 serv
.join(message
[1])
260 self
.chanlist
.append(message
[1])
261 serv
.privmsg(auteur
,"Channels : "+" ".join(self
.chanlist
))
262 log(self
.serveur
,"priv",auteur
," ".join(message
))
264 serv
.privmsg(auteur
,"Channels : "+" ".join(self
.chanlist
))
268 if auteur
in self
.ops
and len(message
)>1:
269 if message
[1] in self
.chanlist
:
270 if not (message
[1] in self
.stay_channels
) or auteur
in self
.overops
:
271 self
.quitter(message
[1]," ".join(message
[2:]))
272 self
.chanlist
.remove(message
[1])
273 log(self
.serveur
,"priv",auteur
," ".join(message
)+"[successful]")
275 serv
.privmsg(auteur
,random
.choice(config
.leave_fail_messages
))
276 log(self
.serveur
,"priv",auteur
," ".join(message
)+"[failed]")
278 serv
.privmsg(auteur
,"Je ne suis pas sur %s"%(message
[1]))
282 if auteur
in self
.overops
:
284 if message
[1] in self
.stay_channels
:
285 log(self
.serveur
,"priv",auteur
," ".join(message
)+"[failed]")
286 serv
.privmsg(auteur
,"Je stay déjà sur %s."%(message
[1]))
288 log(self
.serveur
,"priv",auteur
," ".join(message
)+"[successful]")
289 self
.stay_channels
.append(message
[1])
290 serv
.privmsg(auteur
,"Stay channels : "+" ".join(self
.stay_channels
))
292 serv
.privmsg(auteur
,"Stay channels : "+" ".join(self
.stay_channels
))
296 if auteur
in self
.overops
:
298 if message
[1] in self
.stay_channels
:
299 log(self
.serveur
,"priv",auteur
," ".join(message
)+"[successful]")
300 self
.stay_channels
.remove(message
[1])
301 serv
.privmsg(auteur
,"Stay channels : "+" ".join(self
.stay_channels
))
303 log(self
.serveur
,"priv",auteur
," ".join(message
)+"[failed]")
304 serv
.privmsg(auteur
,"Je ne stay pas sur %s."%(message
[1]))
309 if auteur
in self
.overops
:
310 log(self
.serveur
,"priv",auteur
," ".join(message
)+"[successful]")
315 if auteur
in self
.ops
:
317 if message
[1] in self
.quiet_channels
:
318 serv
.privmsg(auteur
,"Je me la ferme déjà sur %s"%(message
[1]))
319 log(self
.serveur
,"priv",auteur
," ".join(message
)+"[failed]")
321 self
.quiet_channels
.append(message
[1])
322 serv
.privmsg(auteur
,"Quiet channels : "+" ".join(self
.quiet_channels
))
323 log(self
.serveur
,"priv",auteur
," ".join(message
)+"[successful]")
325 serv
.privmsg(auteur
,"Quiet channels : "+" ".join(self
.quiet_channels
))
329 if auteur
in self
.ops
:
331 if message
[1] in self
.quiet_channels
:
332 self
.quiet_channels
.remove(message
[1])
333 serv
.privmsg(auteur
,"Quiet channels : "+" ".join(self
.quiet_channels
))
334 log(self
.serveur
,"priv",auteur
," ".join(message
)+"[successful]")
336 serv
.privmsg(auteur
,"Je ne me la ferme pas sur %s."%(message
[1]))
337 log(self
.serveur
,"priv",auteur
," ".join(message
)+"[failed]")
341 if auteur
in self
.ops
:
343 log(self
.serveur
,"priv",auteur
," ".join(message
)+"[successful]")
347 if auteur
in self
.overops
and len(message
)>2:
348 serv
.privmsg(message
[1]," ".join(message
[2:]))
349 log(self
.serveur
,"priv",auteur
," ".join(message
))
350 elif len(message
)<=2:
351 serv
.privmsg(auteur
,"Syntaxe : SAY <channel> <message>")
355 if auteur
in self
.overops
and len(message
)>2:
356 serv
.action(message
[1]," ".join(message
[2:]))
357 log(self
.serveur
,"priv",auteur
," ".join(message
))
358 elif len(message
)<=2:
359 serv
.privmsg(auteur
,"Syntaxe : DO <channel> <action>")
363 if auteur
in self
.overops
and len(message
)>2:
364 serv
.kick(message
[1],message
[2]," ".join(message
[3:]))
365 log(self
.serveur
,"priv",auteur
," ".join(message
))
366 elif len(message
)<=2:
367 serv
.privmsg(auteur
,"Syntaxe : KICK <channel> <pseudo> [<raison>]")
371 if auteur
in self
.overops
:
372 serv
.privmsg(auteur
," ".join(self
.ops
))
376 if auteur
in self
.overops
:
377 serv
.privmsg(auteur
," ".join(self
.overops
))
383 serv
.privmsg(auteur
,"Je n'ai pas compris. Essayez HELP…")
385 def on_pubmsg(self
, serv
, ev
):
386 auteur
= irclib
.nm_to_n(ev
.source())
388 message
= ev
.arguments()[0]
390 test
=bot_unicode(message
)
391 except UnicodeBotError
:
392 if not canal
in self
.quiet_channels
and config
.utf8_trigger
:
393 serv
.privmsg(canal
, "%s: %s"%(auteur
,config
.utf8_fail
))
395 pour_moi
,message
=self
.pourmoi(serv
,message
)
396 for (detect
, reason
) in config
.kicking_list
:
397 matching
= detect(message
)
399 if canal
in self
.kick_channels
:
400 serv
.kick(canal
,auteur
,(reason
.format(matching
.groups()[0])).encode("utf8"))
402 if pour_moi
and message
.split()!=[]:
403 cmd
=message
.split()[0].lower()
405 args
=" ".join(message
.split()[1:])
408 if cmd
in ["meurs","die","crève","pends-toi"]:
409 if auteur
in self
.overops
:
410 log(self
.serveur
,canal
,auteur
,message
+"[successful]")
413 serv
.privmsg(canal
,"%s: %s"%(auteur
,random
.choice(config
.quit_fail_messages
)))
414 log(self
.serveur
,canal
,auteur
,message
+"[failed]")
415 elif cmd
== "reload":
416 if auteur
in self
.ops
:
417 log(self
.serveur
, canal
, auteur
, message
+"[successful]")
419 elif cmd
in ["part","leave","dégage","va-t-en"]:
420 if auteur
in self
.ops
and (not (canal
in self
.stay_channels
)
421 or auteur
in self
.overops
):
423 log(self
.serveur
,canal
,auteur
,message
+"[successful]")
424 if canal
in self
.chanlist
:
425 self
.chanlist
.remove(canal
)
427 serv
.privmsg(canal
,"%s: %s"%(auteur
,random
.choice(config
.leave_fail_messages
)))
428 log(self
.serveur
,canal
,auteur
,message
+"[failed]")
430 elif cmd
in ["deviens","pseudo"]:
431 if auteur
in self
.ops
:
434 log(self
.serveur
,canal
,auteur
,message
+"[successful]")
436 elif cmd
in ["meur", "meurt","meurre","meurres"] and not canal
in self
.quiet_channels
:
437 serv
.privmsg(canal
,'%s: Mourir, impératif, 2ème personne du singulier : "meurs" (de rien)'%(auteur))
438 elif cmd
in ["ping"] and not canal
in self
.quiet_channels
:
439 serv
.privmsg(canal
,"%s: pong"%(auteur))
440 # if is_insult(message) and not canal in self.quiet_channels:
441 # if is_not_insult(message):
442 # answer=random.choice(config.compliment_answers)
443 # for ligne in answer.split("\n"):
444 # serv.privmsg(canal,"%s: %s"%(auteur,ligne.encode("utf8")))
446 # answer=random.choice(config.insultes_answers)
447 # for ligne in answer.split("\n"):
448 # serv.privmsg(canal,"%s: %s"%(auteur,ligne.encode("utf8")))
449 # elif is_compliment(message) and not canal in self.quiet_channels:
450 # answer=random.choice(config.compliment_answers)
451 # for ligne in answer.split("\n"):
452 # serv.privmsg(canal,"%s: %s"%(auteur,ligne.encode("utf8")))
453 elif is_tesla(message
) and not canal
in self
.quiet_channels
:
454 l1
,l2
=config
.tesla_answers
,config
.tesla_actions
455 n1
,n2
=len(l1
),len(l2
)
456 i
=random
.randrange(n1
+n2
)
458 serv
.action(canal
,l2
[i
-n1
].encode("utf8"))
460 serv
.privmsg(canal
,"%s: %s"%(auteur
,l1
[i
].encode("utf8")))
461 elif is_tag(message
) and not canal
in self
.quiet_channels
:
462 if auteur
in self
.ops
:
463 action
=random
.choice(config
.tag_actions
)
464 serv
.action(canal
,action
.encode("utf8"))
465 self
.quiet_channels
.append(canal
)
467 answer
=random
.choice(config
.tag_answers
)
468 for ligne
in answer
.split("\n"):
469 serv
.privmsg(canal
,"%s: %s"%(auteur
,ligne
.encode("utf8")))
470 elif is_bonjour(message
) and not canal
in self
.quiet_channels
:
472 answer
=random
.choice(config
.night_answers
)
474 answer
=random
.choice(config
.bonjour_answers
)
476 answer
=random
.choice(config
.bonsoir_answers
)
477 serv
.privmsg(canal
,answer
.format(auteur
).encode("utf8"))
478 elif is_bonne_nuit(message
) and not canal
in self
.quiet_channels
:
479 answer
=random
.choice(config
.bonne_nuit_answers
)
480 serv
.privmsg(canal
,answer
.format(auteur
).encode("utf8"))
482 if not canal
in self
.quiet_channels
:
484 if re
.match((u
"^("+u
"|".join(config
.bonjour_triggers
)
485 +u
")( {}| all| tout le monde|(|à) tous)(\.|( |)!|)$"
486 ).format(mypseudo
).lower(), message
.strip().lower()):
487 answer
=random
.choice(config
.bonjour_answers
)
488 serv
.privmsg(canal
,answer
.format(auteur
).encode("utf8"))
490 def on_action(self
, serv
, ev
):
491 action
= ev
.arguments()[0]
492 auteur
= irclib
.nm_to_n(ev
.source())
493 channel
= ev
.target()
495 test
=bot_unicode(action
)
496 except UnicodeBotError
:
497 if not channel
in self
.quiet_channels
and config
.utf8_trigger
:
498 serv
.privmsg(channel
, "%s: %s"%(auteur
,config
.utf8_fail
))
502 # if is_bad_action_trigger(action,mypseudo) and not channel in self.quiet_channels:
503 # l1,l2=config.bad_action_answers,config.bad_action_actions
504 # n1,n2=len(l1),len(l2)
505 # i=random.randrange(n1+n2)
507 # serv.action(channel,l2[i-n1].format(auteur).encode("utf8"))
509 # serv.privmsg(channel,l1[i].format(auteur).format(auteur).encode("utf8"))
510 # if is_good_action_trigger(action,mypseudo) and not channel in self.quiet_channels:
511 # l1,l2=config.good_action_answers,config.good_action_actions
512 # n1,n2=len(l1),len(l2)
513 # i=random.randrange(n1+n2)
515 # serv.action(channel,l2[i-n1].format(auteur).format(auteur).encode("utf8"))
517 # serv.privmsg(channel,l1[i].format(auteur).format(auteur).encode("utf8"))
519 def on_kick(self
,serv
,ev
):
520 auteur
= irclib
.nm_to_n(ev
.source())
521 channel
= ev
.target()
522 victime
= ev
.arguments()[0]
523 raison
= ev
.arguments()[1]
524 if victime
==self
.nick
:
525 log(self
.serveur
,"%s kické de %s par %s (raison : %s)" %(victime
,channel
,auteur
,raison
))
528 username
= irclib
.nm_to_u(ev
.source()).lower()
530 print channel
, username
531 if channel
== "#déprime" and "peb" in username
or "becue" in username
:
533 serv
.kick(auteur
, "Va abuser de tes droits ailleurs !")
535 def kicker(self
, chan
, pseudo
, raison
=None):
537 raison
= config
.kick_default_reason
538 self
.serv
.kick(chan
,pseudo
,raison
)
540 def quitter(self
,chan
,leave_message
=None):
541 if leave_message
==None:
542 leave_message
=random
.choice(config
.leave_messages
)
543 self
.serv
.part(chan
,message
=leave_message
.encode("utf8"))
546 quit_message
=random
.choice(config
.quit_messages
)
547 self
.die(msg
=quit_message
.encode("utf8"))
550 return self
.serv
.get_nickname()
551 nick
=property(_getnick
)
553 def reload(self
, auteur
=None):
555 if auteur
in [None, "SIGHUP"]:
556 towrite
= "Config reloaded" + " (SIGHUP received)"*(auteur
== "SIGHUP")
557 for to
in config
.report_bugs_to
:
558 self
.serv
.privmsg(to
, towrite
)
559 log(self
.serveur
, towrite
)
561 self
.serv
.privmsg(auteur
,"Config reloaded")
563 def start_as_daemon(self
, outfile
):
564 sys
.stderr
= Logger(outfile
)
568 class Logger(object):
569 """Pour écrire ailleurs que sur stdout"""
570 def __init__(self
, filename
="themis.full.log"):
571 self
.filename
= filename
573 def write(self
, message
):
574 f
= open(self
.filename
, "a")
579 if __name__
=="__main__":
582 print "Usage : themis.py <serveur> [--debug] [--no-output] [--daemon [--pidfile]] [--outfile]"
583 print " --outfile sans --no-output ni --daemon n'a aucun effet"
586 if "--daemon" in sys
.argv
:
587 thisfile
= os
.path
.realpath(__file__
)
588 thisdirectory
= thisfile
.rsplit("/", 1)[0]
589 os
.chdir(thisdirectory
)
593 if "debug" in sys
.argv
or "--debug" in sys
.argv
:
597 if "--no-output" in sys
.argv
or "--daemon" in sys
.argv
:
598 outfile
= "/var/log/bots/themis.full.log"
601 if arg
[0].strip('-') in ["out", "outfile", "logfile"]:
603 sys
.stdout
= Logger(outfile
)
604 serveurs
={"a♡":"acoeur.crans.org","acoeur":"acoeur.crans.org","acoeur.crans.org":"acoeur.crans.org",
605 "irc":"irc.crans.org","crans":"irc.crans.org","irc.crans.org":"irc.crans.org"}
607 serveur
=serveurs
[serveur
]
609 print "Server Unknown : %s"%(serveur)
611 themis
=Themis(serveur
,debug
)
612 # Si on reçoit un SIGHUP, on reload la config
613 def sighup_handler(signum
, frame
):
614 themis
.reload("SIGHUP")
615 signal
.signal(signal
.SIGHUP
, sighup_handler
)
617 child_pid
= os
.fork()
620 themis
.start_as_daemon(outfile
)
622 # on enregistre le pid de themis
623 pidfile
= "/var/run/bots/themis.pid"
626 if arg
[0].strip('-') in ["pidfile"]:
628 f
= open(pidfile
, "w")
629 f
.write("%s\n" % child_pid
)