From 537070ceb9402cc2e29dab58a59f2fc9d677bcca Mon Sep 17 00:00:00 2001 From: Vincent Le Gallic Date: Tue, 1 May 2012 19:55:02 +0200 Subject: [PATCH 1/1] =?utf8?q?Basile=20en=20d=C3=A9p=C3=B4t=20git,=20c'est?= =?utf8?q?=20le=20Bien?= MIME-Version: 1.0 Content-Type: text/plain; charset=utf8 Content-Transfer-Encoding: 8bit --- .gitignore | 31 ++ basile.py | 819 ++++++++++++++++++++++++++++++++++++++++++++++ identities.pickle | 10 + 3 files changed, 860 insertions(+) create mode 100644 .gitignore create mode 100755 basile.py create mode 100644 identities.pickle diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..a57fd2e --- /dev/null +++ b/.gitignore @@ -0,0 +1,31 @@ +# Custom # +################### +*~ + +# Compiled source # +################### +*.pyc + +# Packages # +############ +# it's better to unpack these files and commit the raw source +# git has its own built in compression methods +*.7z +*.dmg +*.gz +*.iso +*.jar +*.rar +*.tar +*.zip + +# Logs # +###################### +*.log + +# OS generated files # +###################### +.DS_Store* +*ehthumbs.db +Icon? +*Thumbs.db diff --git a/basile.py b/basile.py new file mode 100755 index 0000000..6e39914 --- /dev/null +++ b/basile.py @@ -0,0 +1,819 @@ +#!/usr/bin/python +# -*- coding:utf8 -*- + +# Codé par 20-100 le 23/04/12 + +# Un test de bot irc, parce que c'est cool + +import irclib +import ircbot +import threading +import random +import time +import socket, ssl, json +import pickle +import re +import os +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" +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",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"] + +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__ ) +def get_filesize(): + return ex("ls -s %s"%(config_thisfile))[1].split()[0] + +class NKError(Exception): + def __init__(self,msg): + Exception.__init__(self) + self.msg=msg + def __str__(self): + return str(self.msg) + def __unicode__(self): + return unicode(self.msg) + +class NKRefused(NKError): + pass + +class NKHelloFailed(NKError): + pass + +class NKUnknownError(NKError): + pass + +def log(serveur,channel,auteur=None,message=None): + f=open(get_config_logfile(serveur),"a") + if auteur==message==None: + # alors c'est que c'est pas un channel mais juste une ligne de log + chain="%s %s"%(time.strftime("%F %T"),channel) + else: + chain="%s [%s:%s] %s"%(time.strftime("%F %T"),channel,auteur,message) + f.write(chain+"\n") + if config_debug_stdout: + print chain + f.close() + +def connect_NK(): + sock=socket.socket() + try: + # On établit la connexion sur port 4242 + sock.connect(("127.0.0.1",4242)) + # On passe en SSL + sock=ssl.wrap_socket(sock,ca_certs='../keys/ca_.crt') + # On fait un hello + sock.write('hello "Basile"') + # On récupère la réponse du hello + out=sock.read() + out=json.loads(out) + except Exception as exc: + # Si on a foiré quelque part, c'est que le serveur est down + raise NKRefused(str(exc)) + if out["retcode"]==0: + return sock + elif out["retcode"]==11: + raise NKHelloFailed(out["errmsg"]) + else: + raise NKUnknownError(out["errmsg"]) + +def login_NK(username,password,typ="bdd"): + sock=connect_NK() + try: + # Basile a un compte special user + commande='login [%s,%s,"%s"]'%(json.dumps(username),json.dumps(password),typ) + sock.write(commande) + out=sock.read() + except Exception as exc: + # Si on a foiré quelque part, c'est que le serveur est down + raise NKRefused(str(exc)) + # On vérifie ensuite que le login + return json.loads(out),sock + + +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 + +def is_insult(chain,debug=True): + 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) |)" + reg=".*pas %s%s.*"%(middle_regexp,insult_regexp) + if re.match(reg,chain): + return True + else: + return False +def is_perdu(chain): + return is_something(chain,config_perdu) +def is_tag(chain): + return is_something(chain,config_tag) +def is_gros(chain): + return is_something(chain,config_gros) +def is_tesla(chain): + return is_something(chain,config_tesla,avant=u"^",apres=u"$",debug=True) +def is_merci(chain): + return is_something(chain,config_merci) +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=" %s($|\.| |,|;).*"%(pseudo)) +def is_pan(chain): + return re.match(u"^(pan|bim|bang)$",unicode(chain,"utf8").lower().strip()) + + + +class UnicodeBotError(Exception): + 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)) + 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.identities=pickle.load(open("identities.pickle","r")) + self.stay_channels=config_stay_channels + self.quiet_channels=config_quiet_channels + self.last_perdu=0 + + + def new_connection_NK(self,serv,username,password,typ="bdd"): + try: + login_result,sock=login_NK(username,password,typ) + droits,retcode,errmsg=login_result["msg"],login_result["retcode"],login_result["errmsg"] + except NKRefused as exc: + for report in self.report_bugs_to: + serv.privmsg(report,"Le Serveur NK2015 est down.") + return (False,None) + except NKHelloFailed as exc: + for report in self.report_bugs_to: + serv.privmsg(report, + "La version du site utilisée n'est pas supportée par le serveur NK2015.") + return (False,None) + except NKUnknownError as exc: + erreurs=["Une fucking erreur inconnue s'est produite"] + erreurs+=str(exc).split("\n") + for report in self.report_bugs_to: + for err in erreurs: + serv.privmsg(report,err) + return (False,None) + except Exception as exc: + # Exception qui ne vient pas de la communication avec le serveur NK2015 + log(self.serveur,"Erreur dans new_connection_NK\n"+str(exc)) + return (False,None) + if retcode==0: + return (True,sock) + else: + 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)) + time.sleep(0.3) + serv.nick(config_irc_pseudo) + + def on_welcome(self, serv, ev): + self.give_me_my_pseudo(serv) + serv.privmsg("NickServ","identify %s"%(config_irc_password)) + log(self.serveur,"Connected") + if self.debug: + self.chanlist=["#bot"] + for c in self.chanlist: + 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] + 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() + size=len(pseudo) + if message[:size]==pseudo and message[size]==":": + return (True,message[size+1:].lstrip(" ")) + else: + return (False,message) + + def on_privmsg(self, serv, ev): + message=ev.arguments()[0] + auteur = irclib.nm_to_n(ev.source()) + try: + test=bot_unicode(message) + except UnicodeBotError: + serv.privmsg(auteur, + "Euh, tu fais de la merde avec ton encodage là, j'ai failli crasher…") + 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é.""" + helpmsg_ops=""" + JOIN Faire rejoindre un channel + LEAVE Faire quitter un channel + QUIET Se taire sur un chanel + NOQUIET Opposé de QUIET + LOST Perdre sur un chan""" + helpmsg_overops=""" + SAY Fais envoyer un message sur un chan ou à une personne + STAY Ignorera les prochains LEAVE pour un chan + NOSTAY Opposé de STAY + DIE Mourir""" + if len(message)==1: + helpmsg=helpmsg_default + if auteur in self.ops: + helpmsg+=helpmsg_ops + if auteur in self.overops: + helpmsg+=helpmsg_overops + else: + helpmsg=helpdico.get(message[1].lower(),"Commande inconnue.") + 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."%( + self.identities[auteur].encode("utf8"))) + else: + serv.privmsg(auteur,"Je ne connais pas ton pseudo note.") + elif len(message)>=3: + username,password=message[1],unicode(" ".join(message[2:]),"utf8") + success,_=self.new_connection_NK(serv,username,password) + if success: + 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")) + else: + log(self.serveur,"priv",auteur," ".join(message)+"[failed]") + serv.privmsg(auteur,"Mot de passe invalide. (ou serveur down)") + else: + serv.privmsg(auteur,u"Syntaxe : IDENTIFY [ ]") + elif cmd=="drop": + if len(message)>1: + if self.identities.has_key(auteur): + password=" ".join(message[1:]) + success,_=self.new_connection_NK(serv,self.identities[auteur],password) + if success: + del self.identities[auteur] + log(self.serveur,"priv",auteur," ".join(message)+"[successful]") + pickle.dump(self.identities,open("identities.pickle","w")) + serv.privmsg(auteur,"Identité oubliée.") + else: + log(self.serveur,"priv",auteur," ".join(message)+"[failed]") + serv.privmsg(auteur,"Mot de passe invalide. (ou serveur down)") + else: + serv.privmsg(auteur,"Je ne connais pas ton pseudo note.") + else: + serv.privmsg(auteur,"Syntaxe : DROP ") + elif cmd=="join": + if auteur in self.ops: + if len(message)>1: + if message[1] in self.chanlist: + serv.privmsg(auteur,"Je suis déjà sur %s"%(message[1])) + else: + serv.join(message[1]) + self.chanlist.append(message[1]) + serv.privmsg(auteur,"Channels : "+" ".join(self.chanlist)) + log(self.serveur,"priv",auteur," ".join(message)) + else: + serv.privmsg(auteur,"Channels : "+" ".join(self.chanlist)) + else: + notunderstood=True + elif cmd=="leave": + 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.chanlist.remove(message[1]) + log(self.serveur,"priv",auteur," ".join(message)+"[successful]") + else: + serv.privmsg(auteur,"Non, je reste !") + log(self.serveur,"priv",auteur," ".join(message)+"[failed]") + else: + serv.privmsg(auteur,"Je ne suis pas sur %s"%(message[1])) + else: + notunderstood=True + elif cmd=="stay": + if auteur in self.overops: + if len(message)>1: + if message[1] in self.stay_channels: + log(self.serveur,"priv",auteur," ".join(message)+"[failed]") + serv.privmsg(auteur,"Je stay déjà sur %s."%(message[1])) + else: + log(self.serveur,"priv",auteur," ".join(message)+"[successful]") + self.stay_channels.append(message[1]) + serv.privmsg(auteur,"Stay channels : "+" ".join(self.stay_channels)) + else: + serv.privmsg(auteur,"Stay channels : "+" ".join(self.stay_channels)) + else: + notunderstood=True + elif cmd=="nostay": + if auteur in self.overops: + if len(message)>1: + if message[1] in self.stay_channels: + log(self.serveur,"priv",auteur," ".join(message)+"[successful]") + self.stay_channels.remove(message[1]) + serv.privmsg(auteur,"Stay channels : "+" ".join(self.stay_channels)) + else: + log(self.serveur,"priv",auteur," ".join(message)+"[failed]") + serv.privmsg(auteur,"Je ne stay pas sur %s."%(message[1])) + + else: + notunderstood=True + elif cmd=="die": + if auteur in self.overops: + log(self.serveur,"priv",auteur," ".join(message)+"[successful]") + self.die() + else: + notunderstood=True + elif cmd=="quiet": + if auteur in self.ops: + if len(message)>1: + if message[1] in self.quiet_channels: + serv.privmsg(auteur,"Je me la ferme déjà sur %s"%(message[1])) + log(self.serveur,"priv",auteur," ".join(message)+"[failed]") + else: + self.quiet_channels.append(message[1]) + serv.privmsg(auteur,"Quiet channels : "+" ".join(self.quiet_channels)) + log(self.serveur,"priv",auteur," ".join(message)+"[successful]") + else: + serv.privmsg(auteur,"Quiet channels : "+" ".join(self.quiet_channels)) + else: + notunderstood=True + elif cmd=="noquiet": + if auteur in self.ops: + if len(message)>1: + if message[1] in self.quiet_channels: + self.quiet_channels.remove(message[1]) + serv.privmsg(auteur,"Quiet channels : "+" ".join(self.quiet_channels)) + log(self.serveur,"priv",auteur," ".join(message)+"[successful]") + else: + serv.privmsg(auteur,"Je ne me la ferme pas sur %s."%(message[1])) + log(self.serveur,"priv",auteur," ".join(message)+"[failed]") + else: + notunderstood=True + elif cmd=="say": + if auteur in self.overops and len(message)>2: + serv.privmsg(message[1]," ".join(message[2:])) + log(self.serveur,"priv",auteur," ".join(message)) + elif len(message)<=2: + serv.privmsg(auteur,"Syntaxe : SAY ") + else: + notunderstood=True + elif cmd=="lost": + if auteur in self.ops and len(message)>1: + serv.privmsg(message[1],"J'ai perdu !") + log(self.serveur,"priv",auteur," ".join(message)) + elif len(message)<=1: + serv.privmsg(auteur,"Syntaxe : LOST ") + else: + notunderstood=True + else: + notunderstood=True + if notunderstood: + serv.privmsg(auteur,"Je n'ai pas compris. Essaye HELP…") + + def on_pubmsg(self, serv, ev): + auteur = irclib.nm_to_n(ev.source()) + canal = ev.target() + message = ev.arguments()[0] + try: + test=bot_unicode(message) + except UnicodeBotError: + serv.privmsg(canal, + "%s: Euh, tu fais de la merde avec ton encodage là, j'ai failli crasher…"%(auteur)) + return + pour_moi,message=self.pourmoi(serv,message) + if pour_moi and message.split()!=[]: + cmd=message.split()[0].lower() + try: + args=" ".join(message.split()[1:]) + except: + args="" + if cmd in ["meurs","die","crève"]: + if auteur in self.overops: + log(self.serveur,canal,auteur,message+"[successful]") + self.die() + else: + serv.privmsg(canal,"%s: crève !"%(auteur)) + log(self.serveur,canal,auteur,message+"[failed]") + + elif cmd in ["part","leave","dégage"]: + 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)) + log(self.serveur,canal,auteur,message+"[successful]") + else: + serv.privmsg(canal,"%s: Non, je reste !"%(auteur)) + 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)[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(canal,"%s: done"%(auteur)) + log(self.serveur,canal,auteur,message+"[successful]") + else: + 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 ?") + else: + serv.privmsg(canal,"%s: crève !"%(auteur)) + log(self.serveur,canal,auteur,message+"[failed]") + + elif cmd in ["deviens","pseudo"]: + if auteur in self.ops: + become=args + 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)) + elif cmd in ["ping"] and not canal in self.quiet_channels: + serv.privmsg(canal,"%s: pong"%(auteur)) + + elif cmd in ["solde","!solde"]: + if self.identities.has_key(auteur): + pseudo=self.identities[auteur] + try: + self.nk.write('search ["x",["pseudo"],%s]'%(json.dumps(pseudo))) + ret=json.loads(self.nk.read()) + solde=ret["msg"][0]["solde"] + pseudo=ret["msg"][0]["pseudo"] + except Exception as exc: + serv.privmsg(canal,"%s: failed"%(auteur)) + log(self.serveur,canal,auteur,message+"[failed]") + else: + 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)) + log(self.serveur,canal,auteur,message+"[unknown]") + elif message in ["pain au chocolat","chocolatine"] 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]": + serv.action(canal,"sert une bouteille de manzana à %s"%(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) + for ligne in answer.split("\n"): + serv.privmsg(canal,"%s: %s"%(auteur,ligne.encode("utf8"))) + else: + 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: + taille=get_filesize() + answer=u"Mais non, je ne suis pas gros, %sKo tout au plus…"%(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 + n1,n2=len(l1),len(l2) + i=random.randrange(n1+n2) + if i>=n1: + serv.action(canal,l2[i-n1]) + 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) + 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) + serv.action(canal,action.encode("utf8")) + self.quiet_channels.append(canal) + else: + 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) + for ligne in answer.split("\n"): + serv.privmsg(canal,"%s: %s"%(auteur,ligne.encode("utf8"))) + out=re.match(u"^([A-Z[]|\\|[0-9]+|(¹|²|³|⁴|⁵|⁶|⁷|⁸|⁹|⁰)+)(?:| \?| !)$", + unicode(message.upper(),"utf8")) + 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 + 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)) + 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): + 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) + 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)) + 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]": + serv.action(canal,"sert une bouteille de manzana à %s"%(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 + answer=random.choice(failanswers) + serv.privmsg(canal,"%s: %s"%(auteur,answer)) + 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) + 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) + if random.uniform(0,1)=n1: + serv.action(channel,l2[i-n1]) + else: + serv.privmsg(channel,"%s: %s"%(auteur,l1[i].encode("utf8"))) + +if __name__=="__main__": + import sys + if len(sys.argv)==1: + print "Usage : basile.py [--debug]" + exit(1) + serveur=sys.argv[1] + if "debug" in sys.argv or "--debug" in sys.argv: + debug=True + else: + debug=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"} + try: + serveur=serveurs[serveur] + except KeyError: + print "Server Unknown : %s"%(serveur) + exit(404) + basile=Basile(serveur,debug) + basile.start() diff --git a/identities.pickle b/identities.pickle new file mode 100644 index 0000000..fa6eaea --- /dev/null +++ b/identities.pickle @@ -0,0 +1,10 @@ +(dp0 +S'[20-100]' +p1 +S'20-100' +p2 +sS'PEB' +p3 +S'peb' +p4 +s. \ No newline at end of file -- 2.39.2