X-Git-Url: http://gitweb.pimeys.fr/?p=bots%2Fbasile.git;a=blobdiff_plain;f=basile.py;h=610a34609e65d374d875e1a315a1ef8a97965e4b;hp=b881878aaf1434fbfccb95ca662703bbd0c6d1f9;hb=9fd92669fd83e2ac4c57874aad0996ea2d0964f9;hpb=4459cc85415618591bbeefed2624ea3c14e6da79 diff --git a/basile.py b/basile.py index b881878..610a346 100755 --- a/basile.py +++ b/basile.py @@ -1,12 +1,10 @@ #!/usr/bin/python # -*- coding:utf8 -*- -# Codé par 20-100 le 23/04/12 +# Codé par 20-100 (commencé le 23/04/12) -# Un test de bot irc, parce que c'est cool +# Un bot IRC qui, un jour, s'interfacera avec la Note Kfet 2015 -import irclib -import ircbot import threading import random import time @@ -14,116 +12,30 @@ import socket, ssl, json import pickle import re import os +import signal +import sys + +# Oui, j'ai recodé ma version d'irclib pour pouvoir rattrapper les SIGHUP +sys.path.insert(0, "/home/vincent/scripts/python-myirclib") +import irclib +import ircbot + from commands import getstatusoutput as ex -import sys -config_debug_stdout=True -if "--quiet" in sys.argv: - config_debug_stdout=False - -config_irc_password="NK2015BasileB0t" -config_irc_pseudo="Basile" -config_chanlist=["#bot","#flood"] -config_stay_channels=["#bot","#flood"] -config_quiet_channels=[] -config_note_pseudo="Basile" -config_note_password="NK2015BasileB0tr4nd0omp4assword]6_+{#]78{" -config_logfile_template="basile.%s.log" +# on récupère la config +import config + +# la partie qui réfère au fichier lui-même est mieux ici +# sinon on réfère la config et pas le fichier lui-même +import os +config.thisfile= os.path.realpath( __file__ ) + def get_config_logfile(serveur): serveurs={"acoeur.crans.org":"acoeur","irc.crans.org":"crans"} - return config_logfile_template%(serveurs[serveur]) -config_overops=["[20-100]","[20-100]_"] -config_ops=["PEB","Nit"] -config_report_bugs_to=["[20-100]"] - -config_insultes=[u"conna(rd|sse)",u"pute",u"con(|ne)",u"enf(oiré|lure)", -u"sal(op(|e(|rie)|ard)|aud)",u"p(e|')tite bite",u"imbécile",u"idiot",u"stupid(|e)",u"débile",u"crétin", -u"pétasse",u"enculé",u"chagasse",u"cagole",u"abruti",u"ahuri",u"analphabète",u"andouille", -u"atardé",u"avorton",u"bachibouzouk",u"(balais|brosse) (de|à) chiotte(|s)", -u"batard",u"blaireau",u"bouffon",u"branque",u"bouseux",u"branleur",u"catin",u"chacal", -u"charogne",u"chiant(|e)",u"chieur",u"cochon",u"coprophage",u"couillon",u"crapule",u"crevard", -u"cruche",u"cuistre",u"ducon",u"décérébré", -u"emmerdeur",u"feignasse",u"fainéant",u"fourbe",u"freluquet",u"frigide", -u"garce",u"glandu",u"gogol",u"goujat",u"gourdasse",u"gredin",u"gringalet",u"grognasse", -u"naze",u"truie",u"iconoclaste", -u"peigne(-|)cul",u"ignare",u"illétré",u"lèche(|-)cul",u"malotru",u"motherfucker",u"nabot",u"nigaud", -u"nul",u"escroc",u"pouffiasse",u"pourriture",u"raclure",u"relou",u"sagouin",u"putain", -u"péripatéticienne"] -config_insultes_answers=[u"toi-même", -u"Oh non ! Quelle insulte ! Je crois que je ne m'en reléverai jamais…\nAh si, ça y est.", -u"J'entends comme un vague murmure, tu disais ?", -u"Je vais prendre ça pour un compliment.", -u"Tu sais, pour toi c'est peut-être une insulte, mais pour moi ce n'est qu'une suite de 0 et de 1…", -u"Si tu allais voir sur un autre chan si j'y suis ?", -u"Permets-moi de te retourner le compliment.", -u"Mais je ne te permets pas !"] - -config_gros=[u"gros"] - -config_buffer_fail_answers=["haha !","You type like you drive","encore un effort ;)"] - -config_premier_groupe_terminaisons=u"(e|es|ons|ez|ent|er(|ai|as|a|ons|ez|ont)|(|er)(ais|ait|ions|iez|aient)|(a(i|s|)|â(mes|tes|t)|èrent)|ass(e(|s|nt)|i(ons|ez))|é(|s|e|es))" -config_regexp_etre=u"(être|suis|e(s|t)|so(mmes|nt)|êtes|(ét|ser)(ai(s|t|ent)|i(ons|ent)|)|ser(ai|as|a|ons|ez|ont)|so(i(s|t|ent)|y(ons|ez))|f(u(s|t|rent)|û(mes|tes|t))|fuss(e(|s|nt)|i(ons|ez))|étant)" -config_regexp_etre_avec_c=u"c'(e(s|st)|étai(t|ent))" -config_regexp_faire=u"fais" -config_perdu=[u"perd(|s|ons|ez|ent|r(e|ai|as|a|ons|ez|ont)|(|r)(ais|ait|ions|iez|aient))" -u"perd(i(s|t|rent)|î(mes|tes|t))", # oui, j'ai inclus qu'il perdît -u"perdiss(e(|s|nt)|i(ons|ez))", -u"perdu(|s|e|es)",u"perdant(|s|e|es)",u"perte(|s)", - -u"(gagn|trouv)"+config_premier_groupe_terminaisons,u"gagnant(|s|e|es)",u"gain(|s)", - -u"trouvant",u"trouvaille(|s)", - -u"victoire(|s)",u"vaincu(|s|e|es)", -u"loose",u"lost",u"looser(|s)",u"win(|ner)(|s)", -u"jeu(|x)",u"game(|s)"] -config_time_between_perdu_trigger=3600*3 #temps moyen pour perdre en l'absence de trigger -config_time_between_perdu_trigger_delta = 30*60 #marge autorisée autour de ^^^ -config_time_between_perdu=30*60 #temps pendant lequel on ne peut pas perdre - -config_tag=[u"t(|a)g",u"ta gueule",u"la ferme",u"ferme( |-)la",u"tais-toi",u"chut"] -config_tag_actions=[u"se tait",u"ferme sa gueule",u"se la ferme",u"la ferme"] -config_tag_answers=[u"J'me tais si j'veux !", -u"Je t'entends pas :°", -u"Héhé, try again", -u"Non, j'ai pas envie", -u"Peut-être quand toi tu la fermeras, et encore…"] - -config_tesla=[u"t('|u )es là \?",u"\?",u"plop \?",u"plouf \?"] -config_tesla_answers=[u"Oui, je suis là",u"Oui ?",u"En quoi puis-je me rendre utile ?"] -config_tesla_actions=[u"est là",u"attend des instructions",u"is alive"] - -config_compliment=[u"gentil",u"cool",u"sympa"] -config_compliment_answers=[u"Merci, c'est gentil :)",u"Je te retourne le compliment",u"C'est gentil ça."] - -config_merci=[u"merci",u"remercie",u"thx",u"thank(|s)"] -config_merci_answers=[u"Mais de rien.",u"À ton service ;)",u"Quand tu veux ^^", -u"Tout le plaisir est pour moi."] - -config_tamere=[u"ta mère"] -config_tamere_answers=[u"Laisse ma mère en dehors de ça !", -u"Tu veux qu'on parle de ta soœur ?", -u"Et la tienne ?", -u"Ce que fait ma mère c'est comme ce que tu fais avec ta bite, ça nous regarde pas…", -u"♩ J'ai vu ta mère sur chat rouleeeeeeette ♫" -u"On avait dit \"pas les mamans\""] - -config_action_trigger=[u"(frappe|cogne|tape)(| sur)",u"(démolit|dégomme|fouette|agresse)", -u"vomit sur",u"slap(|s)"] -config_action_answers=[u"Hey ! Mais qu'est-ce que j'ai fait ?", -u"Pourquoi moi ?", -u"Mais euh…", -u"Mais j'ai rien demandé moi !"] -config_action_actions=[u"prend de la distance, par précaution…",u"part en courant",u"esquive"] - -config_bonjour=[u"(s|)(a|'|)lu(t|)",u"hello",u"plop",u"plip",u"pr(ou|ü)t",u"bonjour",u"bonsoir"] -config_bonjour_answers=[u"Salut {}",u"Hello {} :)",u"Bonjour {}",u"Hello {}",u"{}: hello",u"{}: bonjour"] - - -config_thisfile= os.path.realpath( __file__ ) + return config.logfile_template%(serveurs[serveur]) + def get_filesize(): - return ex("ls -s %s"%(config_thisfile))[1].split()[0] + return ex("ls -s %s"%(config.thisfile))[1].split()[0] class NKError(Exception): def __init__(self,msg): @@ -151,7 +63,7 @@ def log(serveur,channel,auteur=None,message=None): else: chain="%s [%s:%s] %s"%(time.strftime("%F %T"),channel,auteur,message) f.write(chain+"\n") - if config_debug_stdout: + if config.debug_stdout: print chain f.close() @@ -180,9 +92,9 @@ def connect_NK(): def login_NK(username,password,typ="bdd"): sock=connect_NK() if typ=="special": # ça c'est pour Basile lui-même - masque='["note"]' + masque='[]' elif typ=="bdd": - masque='[["all"],["all"],false]' + masque='[[], [], true]' try: # Basile a un compte special user commande='login [%s,%s,"%s",%s]'%(json.dumps(username),json.dumps(password),typ,masque) @@ -195,71 +107,91 @@ def login_NK(username,password,typ="bdd"): return json.loads(out),sock -def is_something(chain,matches,avant=u".*(^| )",apres=u"($|\.| |,|;).*",case_sensitive=False,debug=False): +def is_something(chain,matches,avant=u".*(?:^| )",apres=u"(?:$|\.| |,|;).*",case_sensitive=False,debug=False): if case_sensitive: chain=unicode(chain,"utf8") else: chain=unicode(chain,"utf8").lower() allmatches="("+"|".join(matches)+")" reg=(avant+allmatches+apres).lower() - if re.match(reg,chain): - return True - return False + o=re.match(reg,chain) + return o def is_insult(chain,debug=True): - return is_something(chain,config_insultes,avant=".*(^| |')") + return is_something(chain,config.insultes,avant=".*(?:^| |')") def is_not_insult(chain): chain=unicode(chain,"utf8") - insult_regexp=u"("+u"|".join(config_insultes)+u")" - middle_regexp=u"(un(|e) ((putain|enfoiré) d(e |'))*|)(| super )( (gros|petit|grand|énorme) |)" + insult_regexp=u"("+u"|".join(config.insultes)+u")" + middle_regexp=u"(une? (?:(?:putain|enfoiré) d(?:e |'))*|)(?:| super )(?: (?:gros|petit|grand|énorme) |)" reg=".*pas %s%s.*"%(middle_regexp,insult_regexp) if re.match(reg,chain): return True else: return False +def is_compliment(chain,debug=True): + return is_something(chain,config.compliment_triggers,avant=".*(?:^| |')") def is_perdu(chain): - return is_something(chain,config_perdu) + return is_something(chain,config.perdu) def is_tag(chain): - return is_something(chain,config_tag) + return is_something(chain,config.tag_triggers) def is_gros(chain): - return is_something(chain,config_gros) + return is_something(chain,config.gros) def is_tesla(chain): - return is_something(chain,config_tesla,avant=u"^",apres=u"$",debug=True) + return is_something(chain,config.tesla_triggers,avant=u"^",apres=u"$",debug=True) def is_merci(chain): - return is_something(chain,config_merci) + return is_something(chain,config.merci_triggers) def is_tamere(chain): - return is_something(chain,config_tamere) -def is_action_trigger(chain,pseudo): - return is_something(chain,config_action_trigger,avant=u"^", - apres="( [a-z]*ment)? %s($|\.| |,|;).*"%(pseudo)) + return is_something(chain,config.tamere_triggers) +def is_bad_action_trigger(chain,pseudo): + return is_something(chain,config.bad_action_triggers,avant=u"^", + apres="(?: [a-z]*ment)? %s($|\.| |,|;).*"%(pseudo)) +def is_good_action_trigger(chain,pseudo): + return is_something(chain,config.good_action_triggers,avant=u"^", + apres="(?: [a-z]*ment)? %s($|\.| |,|;).*"%(pseudo)) +def is_bonjour(chain): + return is_something(chain,config.bonjour_triggers,avant=u"^") +def is_bonne_nuit(chain): + return is_something(chain,config.bonne_nuit_triggers,avant=u"^") def is_pan(chain): - return re.match(u"^(pan|bim|bang)$",unicode(chain,"utf8").lower().strip()) + return re.match(u"^(pan|bim|bang)( .*)?$",unicode(chain,"utf8").lower().strip()) +def is_time(conf): + _,_,_,h,m,s,_,_,_=time.localtime() + return (conf[0],0,0)<(h,m,s)<(conf[1],0,0) +def is_day(): + return is_time(config.daytime) +def is_night(): + return is_time(config.nighttime) class UnicodeBotError(Exception): pass + +class CrashError(Exception): + """Pour pouvoir faire crasher Basile, parce que ça a l'air drôle""" + pass + def bot_unicode(chain): try: unicode(chain,"utf8") except UnicodeDecodeError as exc: raise UnicodeBotError + class Basile(ircbot.SingleServerIRCBot): def __init__(self,serveur,debug=False): - temporary_pseudo=config_irc_pseudo+str(random.randrange(10000,100000)) + temporary_pseudo=config.irc_pseudo+str(random.randrange(10000,100000)) ircbot.SingleServerIRCBot.__init__(self, [(serveur, 6667)], temporary_pseudo,"Basile, le bot irc.[Codé par 20-100, fouettez-le]", 10) self.debug=debug self.serveur=serveur - self.overops=config_overops - self.ops=self.overops+config_ops - self.report_bugs_to=config_report_bugs_to - self.chanlist=config_chanlist - self.sockets={} + self.overops=config.overops + self.ops=self.overops+config.ops + self.report_bugs_to=config.report_bugs_to + self.chanlist=config.chanlist self.identities=pickle.load(open("identities.pickle","r")) - self.stay_channels=config_stay_channels - self.quiet_channels=config_quiet_channels + self.stay_channels=config.stay_channels + self.quiet_channels=config.quiet_channels self.last_perdu=0 @@ -293,14 +225,15 @@ class Basile(ircbot.SingleServerIRCBot): return (False,None) def give_me_my_pseudo(self,serv): - serv.privmsg("NickServ","RECOVER %s %s"%(config_irc_pseudo,config_irc_password)) - serv.privmsg("NickServ","RELEASE %s %s"%(config_irc_pseudo,config_irc_password)) + serv.privmsg("NickServ","RECOVER %s %s"%(config.irc_pseudo,config.irc_password)) + serv.privmsg("NickServ","RELEASE %s %s"%(config.irc_pseudo,config.irc_password)) time.sleep(0.3) - serv.nick(config_irc_pseudo) + serv.nick(config.irc_pseudo) def on_welcome(self, serv, ev): + self.serv=serv # ça serv ira :) self.give_me_my_pseudo(serv) - serv.privmsg("NickServ","identify %s"%(config_irc_password)) + serv.privmsg("NickServ","identify %s"%(config.irc_password)) log(self.serveur,"Connected") if self.debug: self.chanlist=["#bot"] @@ -308,82 +241,23 @@ class Basile(ircbot.SingleServerIRCBot): log(self.serveur,"JOIN %s"%(c)) serv.join(c) # on ouvre la connexion note de Basile, special user - self.nk=self.new_connection_NK(serv,config_note_pseudo,config_note_password,"special")[1] + self.nk=self.new_connection_NK(serv,config.note_pseudo,config.note_password,"special")[1] if self.nk==None: for report in self.report_bugs_to: serv.privmsg(report,"Connection to NK2015 failed, invalid password ?") def lost(self,serv,channel,forced=False): - if self.last_perdu+config_time_between_perdu"+u"|".join(adjectifs.keys())+u")" - reg=u".*(^| |')"+debuts+u" "+adj_reg+u"($|,|;|\.| ).*" - matched=re.match(reg,message) - if matched: - # il faut repasser l'adjectif au féminin singulier - found=matched.groupdict()["adjectif"] - for adj in adjectifs.keys(): - if re.match(adj,found): - adjectif=adjectifs[adj] - break - serv.privmsg(channel,(u"%s: c'est ta mère qui est %s !"%(auteur,adjectif)).encode("utf8")) - elif random.randrange(5)==0: - # deuxième type de trigger, mais moins probable - matched=re.match(adj_reg,message) - if matched: - found=matched.groupdict()["adjectif"] - for adj in adjectifs.keys(): - if re.match(adj,found): - adjectif=adjectifs[adj] - break - fille=random.choice([u"mère",u"soœur"]) - serv.privmsg(channel,(u"%s: et ta %s, elle est %s ?"% - (auteur,fille,adjectif)).encode("utf8")) - else: - # troisième type de trigger - cpgt=config_premier_groupe_terminaisons - verbes={u"tourn"+cpgt:u"tourne", - u"balad"+cpgt+u" sur le trottoir":u"se balade sur le trottoir", - u"prom(e|è)n"+cpgt+" sur le trottoir":u"se promène sur le trottoir", - u"_srqhbkjjn":""} - vb_reg=u".*(^| )(?P"+"|".join(verbes.keys())+")( |,|;|\.|$)" - matched=re.match(vb_reg,message) - if matched: - found=matched.groupdict()["verbe"] - for vb in verbes.keys(): - if re.match(vb,found): - verbe=verbes[vb] - break - fille=random.choice([u"mère",u"soœur"]) - serv.privmsg(channel,(u"%s: et ta %s, elle %s ?"% - (auteur,fille,verbe)).encode("utf8")) def pourmoi(self, serv, message): """renvoie (False,lemessage) ou (True, le message amputé de "pseudo: ")""" - pseudo=serv.get_nickname() + pseudo=self.nick size=len(pseudo) if message[:size]==pseudo and len(message)>size and message[size]==":": return (True,message[size+1:].lstrip(" ")) @@ -396,75 +270,89 @@ class Basile(ircbot.SingleServerIRCBot): try: test=bot_unicode(message) except UnicodeBotError: - serv.privmsg(auteur, - "Euh, tu fais de la merde avec ton encodage là, j'ai failli crasher…") + if config.utf8_trigger: + serv.privmsg(auteur, random.choice(config.utf8_fail_answers).encode("utf8")) return message=message.split() cmd=message[0].lower() notunderstood=False - if cmd=="connect": - if not len(message) in [2,3]: - serv.privmsg(auteur,"Syntaxe : CONNECT [] ") - return - username=auteur - if len(message)>2: - username=(message[1]) - password=" ".join(message[2:]) - else: - password=" ".join(message[1:]) - success,sock=self.new_connection_NK(serv,username,password) - if success: - self.sockets[username]=sock - serv.privmsg(auteur,"Connection successful") - log(self.serveur,"priv",auteur," ".join(message)+"[successful]") - else: - serv.privmsg(auteur,"Connection failed") - log(self.serveur,"priv",auteur," ".join(message)+"[failed]") - - elif cmd=="help": - helpdico={"connect": """CONNECT [] - Ouvre une connexion au serveur NoteKfet. - Si n'est pas précisé, j'utiliserais l'identité sous laquelle je te connais, ou, à défaut, ton pseudo.""", -"identify": """IDENTIFY - Vérifie le mot de passe et me permet de savoir à l'avenir quel est ton pseudo note kfet. - Sans paramètre, je réponds sous quel pseudo je te connais.""", -"drop":"""DROP - Vérifie le mot de passe et me fait d'oublier ton pseudo note kfet."""} - helpmsg_default="""Liste des commandes : - HELP Affiche de l'aide sur une commande. - CONNECT Ouvre une connection au serveur Note Kfet. - IDENTIFY Me permet de savoir qui tu es sur la note kfet. - DROP Me fait oublier ton identité. - SOLDE Obtenir ton solde""" - helpmsg_ops=""" - JOIN Faire rejoindre un chan - LEAVE Faire quitter un chan - QUIET Se taire sur un chan - NOQUIET Opposé de QUIET - LOST Perdre sur un chan - SOLDE Donner le solde de quelqu'un""" - helpmsg_overops=""" - SAY Fait envoyer un message sur un chan ou à une personne - STAY Ignorera les prochains LEAVE pour un chan - NOSTAY Opposé de STAY - DIE Mourir""" + if cmd=="help": + helpdico={"help":["""HELP + Affiche de l'aide sur la commande""",None,None], +"identify": ["""IDENTIFY + Vérifie le mot de passe et me permet de savoir à l'avenir quel est votre pseudo note kfet. + Sans paramètre, je vous précise sous quel pseudo je vous connais.""",None,None], +"drop":["""DROP + Vérifie le mot de passe et me fait d'oublier votre pseudo note kfet.""",None,None], +"solde": ["""SOLDE + Affiche votre solde, si je connais votre pseudo note kfet.""", + """SOLDE + Affiche le solde de la personne désignée (par son pseudo note).""",None], + "join": [None, """JOIN + Me fait rejoindre le channel""",None], + "leave": [None,"""LEAVE + Me fait quitter le channel (sauf s'il est dans ma stay_list).""",None], + "quiet": [None,"""QUIET + Me rend silencieux sur le channel.""",None], + "noquiet": [None,"""NOQUIET + Me rend la parole sur le channel.""",None], + "lost": [None,"""LOST + Me fait perdre sur le channel.""",None], + "reconnect": [None,"""RECONNECT + Établit à nouveau la connexion avec le serveur NK2015""",None], + "reload": [None,"""RELOAD + Recharge la configuration.""",None], + "say": [None,None,"""SAY + Me fait parler sur le channel."""], + "do": [None,None,"""DO + Me fait faitre une action (/me) sur le channel."""], + "stay": [None,None,"""STAY + Ajoute le channel à ma stay_list."""], + "nostay": [None,None,"""NOSTAY + Retire le channel de ma stay_list."""], + "ops": [None,None,"""OPS + Affiche la liste des ops."""], + "overops": [None,None,"""OVEROPS + Affiche la liste des overops."""], + "kick": [None,None,"""KICK [] + Kicke du channel (Il faut bien entendu que j'y sois op)."""], + "die": [None,None,"""DIE + Me déconnecte du serveur IRC."""], + "crash": [None,None,"""CRASH + Me fait crasher"""] + } + helpmsg_default="Liste des commandes disponibles :\nHELP IDENTIFY DROP SOLDE" + helpmsg_ops=" JOIN LEAVE QUIET NOQUIET LOST RECONNECT RELOAD" + helpmsg_overops=" SAY DO STAY NOSTAY OPS OVEROPS KICK DIE CRASH" + op,overop=auteur in self.ops, auteur in self.overops if len(message)==1: helpmsg=helpmsg_default - if auteur in self.ops: + if op: helpmsg+=helpmsg_ops - if auteur in self.overops: + if overop: helpmsg+=helpmsg_overops else: - helpmsg=helpdico.get(message[1].lower(),"Commande inconnue.") + helpmsgs=helpdico.get(message[1].lower(),["Commande inconnue.",None,None]) + helpmsg=helpmsgs[0] + if op and helpmsgs[1]: + if helpmsg: + helpmsg+="\n"+helpmsgs[1] + else: + helpmsg=helpmsgs[1] + if overop and helpmsgs[2]: + if helpmsg: + helpmsg+="\n"+helpmsgs[2] + else: + helpmsg=helpmsgs[2] for ligne in helpmsg.split("\n"): serv.privmsg(auteur,ligne) elif cmd=="identify": if len(message)==1: if self.identities.has_key(auteur): - serv.privmsg(auteur,"Je te connais sous le pseudo note %s."%( + serv.privmsg(auteur,"Je vous connais sous le pseudo note %s."%( self.identities[auteur].encode("utf8"))) else: - serv.privmsg(auteur,"Je ne connais pas ton pseudo note.") + serv.privmsg(auteur,"Je ne connais pas votre pseudo note.") elif len(message)>=3: username,password=message[1],unicode(" ".join(message[2:]),"utf8") success,_=self.new_connection_NK(serv,username,password) @@ -472,7 +360,7 @@ class Basile(ircbot.SingleServerIRCBot): log(self.serveur,"priv",auteur," ".join(message)+"[successful]") serv.privmsg(auteur,"Identité enregistrée.") self.identities[auteur]=username - pickle.dump(self.identities,open("identities.pickle","w")) + pickle.dump(Xself.identities,open("identities.pickle","w")) else: log(self.serveur,"priv",auteur," ".join(message)+"[failed]") serv.privmsg(auteur,"Mot de passe invalide. (ou serveur down)") @@ -513,7 +401,7 @@ class Basile(ircbot.SingleServerIRCBot): if auteur in self.ops and len(message)>1: if message[1] in self.chanlist: if not (message[1] in self.stay_channels) or auteur in self.overops: - serv.part(message[1]) + self.quitter(message[1]," ".join(message[2:])) self.chanlist.remove(message[1]) log(self.serveur,"priv",auteur," ".join(message)+"[successful]") else: @@ -553,7 +441,37 @@ class Basile(ircbot.SingleServerIRCBot): elif cmd=="die": if auteur in self.overops: log(self.serveur,"priv",auteur," ".join(message)+"[successful]") - self.die() + self.mourir() + else: + notunderstood=True + elif cmd=="crash": + if auteur in self.overops: + log(self.serveur,"priv",auteur," ".join(message)+"[successful]") + self.crash() + else: + notunderstood=True + elif cmd=="reload": + if auteur in self.ops: + self.reload(auteur) + log(self.serveur,"priv",auteur," ".join(message)+"[successful]") + else: + notunderstood=True + elif cmd=="reconnect": + if auteur in self.ops: + try: + self.nk=self.new_connection_NK(serv,config.note_pseudo, + config.note_password,"special")[1] + except Exception as exc: + self.nk=None + log(self.serveur,"""Erreur dans on_pubmsg/"cmd in ["reconnect"]\n"""+str(exc)) + if self.nk!=None: + serv.privmsg(auteur,"%s: done"%(auteur)) + log(self.serveur,"priv",auteur," ".join(message)+"[successful]") + else: + serv.privmsg(auteur,"%s: failed"%(auteur)) + log(self.serveur,"priv",auteur," ".join(message)+"[failed]") + for report in self.report_bugs_to: + serv.privmsg(report,"Connection to NK2015 failed, invalid password ? Server dead ?") else: notunderstood=True elif cmd=="quiet": @@ -590,6 +508,22 @@ class Basile(ircbot.SingleServerIRCBot): serv.privmsg(auteur,"Syntaxe : SAY ") else: notunderstood=True + elif cmd=="do": + if auteur in self.overops and len(message)>2: + serv.action(message[1]," ".join(message[2:])) + log(self.serveur,"priv",auteur," ".join(message)) + elif len(message)<=2: + serv.privmsg(auteur,"Syntaxe : DO ") + else: + notunderstood=True + elif cmd=="kick": + if auteur in self.overops and len(message)>2: + serv.kick(message[1],message[2]," ".join(message[3:])) + log(self.serveur,"priv",auteur," ".join(message)) + elif len(message)<=2: + serv.privmsg(auteur,"Syntaxe : KICK []") + else: + notunderstood=True elif cmd=="lost": if auteur in self.ops and len(message)>1: serv.privmsg(message[1],"J'ai perdu !") @@ -612,6 +546,7 @@ class Basile(ircbot.SingleServerIRCBot): log(self.serveur,"priv",auteur," ".join(message)+"[failed]") return serv.privmsg(auteur,"%s (%s)"%(float(solde)/100,pseudo.encode("utf8"))) + log(self.serveur,"priv",auteur," ".join(message)+"[successful]") else: serv.privmsg(canal,"Je ne connais pas ton pseudo note.") elif auteur in self.ops: @@ -625,10 +560,20 @@ class Basile(ircbot.SingleServerIRCBot): log(self.serveur,"priv",auteur," ".join(message)+"[failed]") return serv.privmsg(auteur,"%s (%s)"%(float(solde)/100,pseudo.encode("utf8"))) + elif cmd=="ops": + if auteur in self.overops: + serv.privmsg(auteur," ".join(self.ops)) + else: + notunderstood=True + elif cmd=="overops": + if auteur in self.overops: + serv.privmsg(auteur," ".join(self.overops)) + else: + notunderstood=True else: notunderstood=True if notunderstood: - serv.privmsg(auteur,"Je n'ai pas compris. Essaye HELP…") + serv.privmsg(auteur,"Je n'ai pas compris. Essayez HELP…") def on_pubmsg(self, serv, ev): auteur = irclib.nm_to_n(ev.source()) @@ -637,10 +582,9 @@ class Basile(ircbot.SingleServerIRCBot): try: test=bot_unicode(message) except UnicodeBotError: - if not canal in self.quiet_channels: - serv.privmsg(canal, - "%s: Euh, tu fais de la merde avec ton encodage là, j'ai failli crasher…"%(auteur)) - return + if config.utf8_trigger and not canal in self.quiet_channels: + serv.privmsg(canal, (u"%s: %s"%(auteur,random.choice(config.utf8_fail_answers))).encode("utf8")) + return pour_moi,message=self.pourmoi(serv,message) if pour_moi and message.split()!=[]: cmd=message.split()[0].lower() @@ -651,25 +595,33 @@ class Basile(ircbot.SingleServerIRCBot): if cmd in ["meurs","die","crève"]: if auteur in self.overops: log(self.serveur,canal,auteur,message+"[successful]") - self.die() + self.mourir() else: - serv.privmsg(canal,"%s: crève !"%(auteur)) + serv.privmsg(canal,("%s: %s"%(auteur,random.choice(config.quit_fail_messages))).encode("utf8")) log(self.serveur,canal,auteur,message+"[failed]") - - elif cmd in ["part","leave","dégage"]: + elif cmd == "reload": + if auteur in self.ops: + log(self.serveur, canal, auteur, message+"[successful]") + self.reload(canal) + elif cmd == "crash": + if auteur in self.overops: + self.crash() + elif cmd in ["part","leave","dégage","va-t-en","tut'tiresailleurs,c'estmesgalets"]: if auteur in self.ops and (not (canal in self.stay_channels) or auteur in self.overops): - serv.part(canal,message="Éjecté par %s"%(auteur)) + self.quitter(canal) log(self.serveur,canal,auteur,message+"[successful]") + if canal in self.chanlist: + self.chanlist.remove(canal) else: - serv.privmsg(canal,"%s: Non, je reste !"%(auteur)) + serv.privmsg(canal,("%s: %s"%(auteur,random.choice(config.leave_fail_messages))).encode("utf8")) log(self.serveur,canal,auteur,message+"[failed]") elif cmd in ["reconnect"]: if auteur in self.ops: try: - self.nk=self.new_connection_NK(serv,config_note_pseudo, - config_note_password,"special")[1] + self.nk=self.new_connection_NK(serv,config.note_pseudo, + config.note_password,"special")[1] except Exception as exc: self.nk=None log(self.serveur,"""Erreur dans on_pubmsg/"cmd in ["reconnect"]\n"""+str(exc)) @@ -680,9 +632,9 @@ class Basile(ircbot.SingleServerIRCBot): serv.privmsg(canal,"%s: failed"%(auteur)) log(self.serveur,canal,auteur,message+"[failed]") for report in self.report_bugs_to: - serv.privmsg(report,"Connection to NK2015 failed, invalid password ?") + serv.privmsg(report,"Connection to NK2015 failed, invalid password ? Server dead ?") else: - serv.privmsg(canal,"%s: crève !"%(auteur)) + serv.privmsg(canal,"%s: %s"%(auteur,random.choice(config.pas_programme_pour_tobeir).encode("utf8"))) log(self.serveur,canal,auteur,message+"[failed]") elif cmd in ["deviens","pseudo"]: @@ -691,8 +643,8 @@ class Basile(ircbot.SingleServerIRCBot): serv.nick(become) log(self.serveur,canal,auteur,message+"[successful]") - elif cmd in ["coucou"] and not canal in self.quiet_channels: - serv.privmsg(canal,"%s: coucou"%(auteur)) + if cmd in ["meur", "meurt","meurre","meurres"] and not canal in self.quiet_channels: + serv.privmsg(canal,'%s: Mourir, impératif, 2ème personne du singulier : "meurs" (de rien)'%(auteur)) elif cmd in ["ping"] and not canal in self.quiet_channels: serv.privmsg(canal,"%s: pong"%(auteur)) @@ -711,114 +663,132 @@ class Basile(ircbot.SingleServerIRCBot): serv.privmsg(canal,"%s: %s (%s)"%(auteur,float(solde)/100,pseudo.encode("utf8"))) log(self.serveur,canal,auteur,message+"[successful]") else: - serv.privmsg(canal,"%s: Je ne connais pas ton pseudo note."%(auteur)) + serv.privmsg(canal,"%s: Je ne connais pas votre pseudo note."%(auteur)) log(self.serveur,canal,auteur,message+"[unknown]") elif (re.match("!?(pain au chocolat|chocolatine)",message.lower()) and not canal in self.quiet_channels): serv.action(canal,"sert un pain au chocolat à %s"%(auteur)) elif re.match("!?manzana",message.lower()) and not canal in self.quiet_channels: - if auteur=="[20-100]": + if auteur in config.manzana: serv.action(canal,"sert une bouteille de manzana à %s"%(auteur)) + elif auteur in config.manzana_bis: + serv.action(canal,"sert un grand verre de jus de pomme à %s : tout le monde sait qu'il ne boit pas."%(auteur)) else: serv.action(canal,"sert un verre de manzana à %s"%(auteur)) if is_insult(message) and not canal in self.quiet_channels: if is_not_insult(message): - answer=random.choice(config_compliment_answers) + answer=random.choice(config.compliment_answers) for ligne in answer.split("\n"): serv.privmsg(canal,"%s: %s"%(auteur,ligne.encode("utf8"))) else: - answer=random.choice(config_insultes_answers) + answer=random.choice(config.insultes_answers) for ligne in answer.split("\n"): serv.privmsg(canal,"%s: %s"%(auteur,ligne.encode("utf8"))) - if is_gros(message) and not canal in self.quiet_channels: + elif is_compliment(message) and not canal in self.quiet_channels: + answer=random.choice(config.compliment_answers) + for ligne in answer.split("\n"): + serv.privmsg(canal,"%s: %s"%(auteur,ligne.encode("utf8"))) + gros_match=is_gros(message) + if gros_match and not canal in self.quiet_channels: taille=get_filesize() - answer=u"Mais non, je ne suis pas gros, %sKo tout au plus…"%(taille) + answer=u"Mais non, je ne suis pas %s, %sKo tout au plus…"%(gros_match.groups()[0],taille) serv.privmsg(canal,"%s: %s"%(auteur,answer.encode("utf8"))) if is_tesla(message) and not canal in self.quiet_channels: - l1,l2=config_tesla_answers,config_tesla_actions + l1,l2=config.tesla_answers,config.tesla_actions n1,n2=len(l1),len(l2) i=random.randrange(n1+n2) if i>=n1: - serv.action(canal,l2[i-n1]) + serv.action(canal,l2[i-n1].encode("utf8")) else: serv.privmsg(canal,"%s: %s"%(auteur,l1[i].encode("utf8"))) if is_tamere(message) and not canal in self.quiet_channels: - answer=random.choice(config_tamere_answers) + answer=random.choice(config.tamere_answers) for ligne in answer.split("\n"): serv.privmsg(canal,"%s: %s"%(auteur,ligne.encode("utf8"))) if is_tag(message) and not canal in self.quiet_channels: if auteur in self.ops: - action=random.choice(config_tag_actions) + action=random.choice(config.tag_actions) serv.action(canal,action.encode("utf8")) self.quiet_channels.append(canal) else: - answer=random.choice(config_tag_answers) + answer=random.choice(config.tag_answers) for ligne in answer.split("\n"): serv.privmsg(canal,"%s: %s"%(auteur,ligne.encode("utf8"))) if is_merci(message): - answer=random.choice(config_merci_answers) + answer=random.choice(config.merci_answers) for ligne in answer.split("\n"): serv.privmsg(canal,"%s: %s"%(auteur,ligne.encode("utf8"))) - out=re.match(u"^([A-Z[]|\\|[0-9]+|(¹|²|³|⁴|⁵|⁶|⁷|⁸|⁹|⁰)+)(?:| \?| !)$", + out=re.match(ur"^([A-Z[]|\\|[0-9]+|(¹|²|³|⁴|⁵|⁶|⁷|⁸|⁹|⁰)+)(?:| \?| !)$", unicode(message.upper(),"utf8")) + if re.match("ma bite dans ton oreille",message) and not canal in self.quiet_channels: + serv.privmsg(canal,"%s: Seul un olasd peut imiter un olasd dans un de ses grands jours !"%(auteur)) if out and not canal in self.quiet_channels: out=out.groups()[0] try: out=int(out) serv.privmsg(canal,"%s: %s !"%(auteur,out+1)) - if out+1>1000 and random.randrange(4)==0: - serv.privmsg(canal,"%s: Tu sais, je peux continuer longtemps comme ça…"%(auteur)) if out==2147483647: - serv.privmsg(canal,"%s: Tu croyais m'avoir sur le maxint ? J'suis en python mon vieux, 'va falloir trouver mieux…"%(auteur)) - return + serv.privmsg(canal,"%s: Ciel, un maxint ! Heureusement que je suis en python…"%(auteur)) + return + if out+1>1000 and random.randrange(4)==0: + serv.privmsg(canal,"%s: Vous savez, moi et les chiffres…"%(auteur)) + return except Exception as exc: pass if re.match("[A-Y]",out): alphabet="ABCDEFGHIJKLMNOPQRSTUVWXYZ" serv.privmsg(canal,"%s: %s !"%(auteur,alphabet[alphabet.index(out)+1])) elif out=="Z": - serv.privmsg(canal,"%s: pfff, j'ai l'air malin maintenant… [ ?"%(auteur)) + serv.privmsg(canal,"%s: Je ne vous remercie pas, j'ai l'air idiot ainsi… [ ?"%(auteur)) elif out in "[\\": - serv.privmsg(canal,"%s: nan mais il faut qu'on arrête, ça va finir par poser des problèmes…"%(auteur)) - elif re.match(r"(¹|²|³|⁴|⁵|⁶|⁷|⁸|⁹|⁰)+",out): + serv.privmsg(canal,"%s: Nous devrions nous en tenir là, ça va finir par poser des problèmes…"%(auteur)) + elif re.match(ur"(¹|²|³|⁴|⁵|⁶|⁷|⁸|⁹|⁰)+",out): def translate(mess): return "".join([{u"⁰¹²³⁴⁵⁶⁷⁸⁹0123456789"[i]:u"0123456789⁰¹²³⁴⁵⁶⁷⁸⁹"[i] for i in range(20)}[j] for j in mess]) out=int(translate(out)) serv.privmsg(canal,"%s: %s !"%(auteur,translate(str(out+1)).encode("utf8"))) - if (not canal in self.quiet_channels - and re.match((u"^("+"|".join(config_bonjour)+").*").lower(),message.lower()) ): - answer=random.choice(config_bonjour_answers) + if is_bonjour(message) and not canal in self.quiet_channels: + if is_night(): + answer=random.choice(config.night_answers) + elif is_day(): + answer=random.choice(config.bonjour_answers) + else: + answer=random.choice(config.bonsoir_answers) + serv.privmsg(canal,answer.format(auteur).encode("utf8")) + if is_bonne_nuit(message) and not canal in self.quiet_channels: + answer=random.choice(config.bonne_nuit_answers) serv.privmsg(canal,answer.format(auteur).encode("utf8")) - if is_pan(message): - serv.privmsg(canal,"%s: c'est pas sur moi qu'il faut tirer !"%(auteur)) + if is_pan(message) and not canal in self.quiet_channels: + serv.privmsg(canal,"%s: ce n'est pas sur moi qu'il faut tirer, même si je sais que j'attire l'œil !"%(auteur)) else: if message in ["!pain au chocolat","!chocolatine"] and not canal in self.quiet_channels: serv.action(canal,"sert un pain au chocolat à %s"%(auteur)) if message in ["!manzana"] and not canal in self.quiet_channels: - if auteur=="[20-100]": + if auteur in config.manzana: serv.action(canal,"sert une bouteille de manzana à %s"%(auteur)) + elif auteur in config.manzana_bis: + serv.action(canal,"sert un grand verre de jus de pomme à %s : tout le monde sait qu'il ne boit pas."%(auteur)) else: serv.action(canal,"sert un verre de manzana à %s"%(auteur)) - if re.match('^(.|§|:|)(w|b) [0-9]+$',message) and not canal in self.quiet_channels: - failanswers=config_buffer_fail_answers + if re.match(u'^ *(.|§|!|/|/|:|)(w|b) [0-9]+$',message.decode("utf8")) and not canal in self.quiet_channels: + failanswers=config.buffer_fail_answers answer=random.choice(failanswers) - serv.privmsg(canal,"%s: %s"%(auteur,answer)) + serv.privmsg(canal,("%s: %s"%(auteur,answer)).encode("utf8")) if not canal in self.quiet_channels: - self.try_tamere(serv,canal,auteur,message) - mypseudo=serv.get_nickname() - if re.match((u"^("+u"|".join(config_bonjour) - +u")( {}| all| tout le monde|(|à) tous)(\.|( |)!|)$" - ).format(mypseudo).lower(), message.strip().lower()): - answer=random.choice(config_bonjour_answers) + mypseudo=self.nick + if re.match((u"^("+u"|".join(config.bonjour_triggers) + +ur")( {}| all| tout le monde| (à )?tous)(\.| ?!)?$" + ).format(mypseudo).lower(), message.decode("utf8").strip().lower()): + answer=random.choice(config.bonjour_answers) serv.privmsg(canal,answer.format(auteur).encode("utf8")) if (is_perdu(message) and not canal in self.quiet_channels): # proba de perdre sur trigger : # avant 30min (enfin, config) : 0 # ensuite, +25%/30min, linéairement deltat=time.time()-self.last_perdu - barre=(deltat-config_time_between_perdu)/(2*3600.0) + barre=(deltat-config.time_between_perdu)/(2*3600.0) if random.uniform(0,1)=n1: - serv.action(channel,l2[i-n1].encode("utf8")) + serv.action(channel,l2[i-n1].format(auteur).encode("utf8")) else: - serv.privmsg(channel,"%s: %s"%(auteur,l1[i].encode("utf8"))) + serv.privmsg(channel,l1[i].format(auteur).format(auteur).encode("utf8")) + if is_good_action_trigger(action,mypseudo) and not channel in self.quiet_channels: + l1,l2=config.good_action_answers,config.good_action_actions + n1,n2=len(l1),len(l2) + i=random.randrange(n1+n2) + if i>=n1: + serv.action(channel,l2[i-n1].format(auteur).format(auteur).encode("utf8")) + else: + serv.privmsg(channel,l1[i].format(auteur).format(auteur).encode("utf8")) + + def on_kick(self,serv,ev): + auteur = irclib.nm_to_n(ev.source()) + channel = ev.target() + victime = ev.arguments()[0] + raison = ev.arguments()[1] + if victime==self.nick: + log(self.serveur,"%s kické de %s par %s (raison : %s)" %(victime,channel,auteur,raison)) + time.sleep(2) + serv.join(channel) + l1,l2=config.kick_answers,config.kick_actions + n1,n2=len(l1),len(l2) + i=random.randrange(n1+n2) + if i>=n1: + serv.action(channel,l2[i-n1].format(auteur).encode("utf8")) + else: + serv.privmsg(channel,l1[i].format(auteur).encode("utf8")) + + def quitter(self,chan,leave_message=None): + if leave_message==None: + leave_message=random.choice(config.leave_messages) + self.serv.part(chan,message=leave_message.encode("utf8")) + + def mourir(self): + quit_message=random.choice(config.quit_messages) + self.die(msg=quit_message.encode("utf8")) + + def _getnick(self): + return self.serv.get_nickname() + nick=property(_getnick) + + def reload(self, auteur=None): + reload(config) + if auteur in [None, "SIGHUP"]: + towrite = "Config reloaded" + " (SIGHUP received)"*(auteur == "SIGHUP") + for to in config.report_bugs_to: + self.serv.privmsg(to, towrite) + log(self.serveur, towrite) + else: + self.serv.privmsg(auteur,"Config reloaded") + + def crash(self): + raise CrashError + + def start_as_daemon(self, outfile): + sys.stderr = Logger(outfile) + self.start() + -if __name__=="__main__": - import sys +class Logger(object): + """Pour écrire ailleurs que sur stdout""" + def __init__(self, filename="basile.full.log"): + self.filename = filename + + def write(self, message): + f = open(self.filename, "a") + f.write(message) + f.close() + +def main(): if len(sys.argv)==1: - print "Usage : basile.py [--debug]" + print "Usage : basile.py [--debug] [--no-output] [--daemon [--pidfile]] [--outfile]" + print " --outfile sans --no-output ni --daemon n'a aucun effet" exit(1) serveur=sys.argv[1] + if "--daemon" in sys.argv: + thisfile = os.path.realpath(__file__) + thisdirectory = thisfile.rsplit("/", 1)[0] + os.chdir(thisdirectory) + daemon = True + else: + daemon = False if "debug" in sys.argv or "--debug" in sys.argv: debug=True else: debug=False + if "--quiet" in sys.argv: + config.debug_stdout=False serveurs={"a♡":"acoeur.crans.org","acoeur":"acoeur.crans.org","acoeur.crans.org":"acoeur.crans.org", "irc":"irc.crans.org","crans":"irc.crans.org","irc.crans.org":"irc.crans.org"} + if "--no-output" in sys.argv or "--daemon" in sys.argv: + outfile = "/var/log/bots/basile.full.log" + for arg in sys.argv: + arg = arg.split("=") + if arg[0].strip('-') in ["out", "outfile", "logfile"]: + outfile = arg[1] + sys.stdout = Logger(outfile) try: serveur=serveurs[serveur] except KeyError: print "Server Unknown : %s"%(serveur) exit(404) basile=Basile(serveur,debug) - basile.start() + # Si on reçoit un SIGHUP, on reload la config + def sighup_handler(signum, frame): + basile.reload("SIGHUP") + signal.signal(signal.SIGHUP, sighup_handler) + if daemon: + child_pid = os.fork() + if child_pid == 0: + os.setsid() + basile.start_as_daemon(outfile) + else: + # on enregistre le pid de basile + pidfile = "/var/run/bots/basile.pid" + for arg in sys.argv: + arg = arg.split("=") + if arg[0].strip('-') in ["pidfile"]: + pidfile = arg[1] + f = open(pidfile, "w") + f.write("%s\n" % child_pid) + f.close() + else: + basile.start() + +if __name__ == "__main__": + main()