]> gitweb.pimeys.fr Git - bots/basile.git/commitdiff
Basile en dépôt git, c'est le Bien
authorVincent Le Gallic <legallic@crans.org>
Tue, 1 May 2012 17:55:02 +0000 (19:55 +0200)
committerVincent Le Gallic <legallic@crans.org>
Tue, 1 May 2012 17:55:02 +0000 (19:55 +0200)
.gitignore [new file with mode: 0644]
basile.py [new file with mode: 0755]
identities.pickle [new file with mode: 0644]

diff --git a/.gitignore b/.gitignore
new file mode 100644 (file)
index 0000000..a57fd2e
--- /dev/null
@@ -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 (executable)
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<time.time() or forced:
+            if not channel in self.quiet_channels or forced:
+                serv.privmsg(channel,"J'ai perdu !")
+            self.last_perdu=time.time()
+            delay=config_time_between_perdu_trigger
+            delta=config_time_between_perdu_trigger_delta
+            serv.execute_delayed(random.randrange(delay-delta,delay+delta),self.lost,(serv,channel))
+    
+    def try_tamere(self,serv,channel,auteur,message):
+        """Essaye de trigger un ta mère"""
+        #pas à chaque fois quand même
+        if random.randrange(4)==0:
+            debuts=u"("+config_regexp_etre+u"|"+config_regexp_etre_avec_c+u")"
+            adjectifs={u"bon(|ne|s|nes)":u"bonne",
+                       u"baisable(|s)":u"baisable",
+                       u"faisable(|s)":u"faisable",
+                       u"pas ch(ère(|s)|er(|s))":u"pas chère",
+                       u"facile(|s)":u"facile",
+                       u"chaud(|e|s|es)":u"chaude",
+                       u"gratuit(|e|s|es)":u"gratuite",
+                       u"payant(|e|s|es)":u"payante",
+                       u"ouvert(|e|s|es)":u"ouverte",
+                       u"open":u"open",
+                       u"plein(|s|es)":u"pleine",
+                       u"bien plein(|e|s|es)":u"bien pleine"}
+            adj_reg=u"(?P<adjectif>"+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<verbe>"+"|".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 [<username>] <password>")
+                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 [<username>] <password>
+ Ouvre une connexion au serveur NoteKfet.
+ Si <username> n'est pas précisé, j'utiliserais l'identité sous laquelle je te connais, ou, à défaut, ton pseudo.""", 
+"identify": """IDENTIFY <username> <password>
+ 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 <password>
+ 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 [<username> <password>]")
+        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 <password>")
+        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 <channel> <message>")
+            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 <channel>")
+            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)<barre:
+                serv.privmsg(canal,"%s: J'ai perdu !"%(auteur))
+                self.last_perdu=time.time()
+
+    def on_action(self, serv, ev):
+        action = ev.arguments()[0]
+        auteur = irclib.nm_to_n(ev.source())
+        channel = ev.target()
+        mypseudo=serv.get_nickname()
+        if is_action_trigger(action,mypseudo):
+            l1,l2=config_action_answers,config_action_actions
+            n1,n2=len(l1),len(l2)
+            i=random.randrange(n1+n2)
+            if i>=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 <serveur> [--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 (file)
index 0000000..fa6eaea
--- /dev/null
@@ -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