2 # -*- encoding: utf-8 -*-
4 """ Gestion des quotes """
14 quote_matcher
= re
.compile(config
.quote_regexp
, flags
=re
.UNICODE
)
15 quote_matcher_with_timestamp
= re
.compile(config
.quote_regexp_with_timestamp
, flags
=re
.UNICODE
)
16 spaces_matcher
= re
.compile(u
"\s", flags
=re
.U
)
18 def equivalence_partition(iterable
, relation
):
19 """ Partitionne l'itérable en classes d'équivalences. """
22 # find the class it is in
25 if relation( iter(c
).next(), o
): # is it equivalent to this class?
29 if not found
: # it is in a new class
30 classes
.append( set( [ o
] ) )
34 """ Renvoie la date actuelle """
35 return datetime
.datetime(*time
.localtime()[:6])
37 def sanitize_author(raw
):
38 """Proprifie l'auteur : enlève les espaces insécables."""
39 return spaces_matcher
.sub(u
" ", raw
)
43 def __init__(self
, author
, content
, timestamp
=None):
46 elif isinstance(timestamp
, basestring
):
47 timestamp
= datetime
.datetime(*time
.strptime(timestamp
, u
"%Y-%m-%d_%H:%M:%S")[:6])
48 self
.author
= sanitize_author(author
)
49 self
.content
= content
50 self
.timestamp
= timestamp
53 d
= {"author" : self
.author
, "content" : self
.content
,
54 "timestamp" : self
.timestamp
.strftime(u
"%F_%T")}
57 def __unicode__(self
):
58 """ Retourne la quote affichable """
59 return config
.quote_template
% self
.__dict
__
61 return unicode(self
).encode("utf-8")
63 def __eq__(self
, otherquote
):
64 """ Vérifie si cette phrase n'a pas déjà été dite par la même personne.
65 Indépendamment de la date et de la casse. """
66 return [self
.author
.lower(), self
.content
.lower()] == [otherquote
.author
.lower(), otherquote
.content
.lower()]
69 def parse(text
, date
=None):
70 """ Parse le ``text`` et renvoie une quote ou None. """
73 get
= quote_matcher
.match(text
)
76 return Quote(d
["author"], d
["content"], date
)
78 def load_file(filename
):
79 """ Récupère les quotes depuis le fichier """
80 with
open(filename
) as f
:
81 jsonquotes
= json
.load(f
)
82 quotes
= [Quote(**q
) for q
in jsonquotes
]
85 def save_file(quotes
, filename
):
86 """ Enregistre les quotes dans le fichier """
87 with
open(filename
, "w") as f
:
88 raws
= [q
.jsonize() for q
in quotes
]
91 class QuoteDB(object):
92 """ Stocke et distribue des quotes. """
97 """ Charge le fichier de quotes dans la DB """
98 self
.quotelist
= load_file(config
.quote_file
)
101 """ Sauvegarde la DB dans le fichier de quotes """
102 save_file(self
.quotelist
, config
.quote_file
)
104 def _collapse_author(self
, author
):
105 """ Renvoie ``author`` avec la casse déjà utilisée si il a déjà été quoté
106 sinon, le renvoie sans le modifier. """
107 authors
= list(set([q
.author
for q
in self
.quotelist
if q
.author
.lower() == author
.lower()]))
109 print "Warning : authors %s" % authors
115 def get_clash_authors(self
):
116 """ Renvoie une liste de liste d'auteurs qui sont enresgitrés avec des casses différentes. """
117 authors
= list(set([q
.author
for q
in self
.quotelist
]))
118 authors
= equivalence_partition(authors
, lambda x
,y
: x
.lower() == y
.lower())
119 authors
= [list(c
) for c
in authors
if len(c
) > 1]
122 def store(self
, author
, content
, timestamp
=None):
123 """ Enregistre une nouvelle quote, sauf si elle existe déjà.
124 Force l'auteur à utiliser la même casse si un auteur de casse différente existait déjà.
125 Renvoie ``True`` si elle a été ajoutée, ``False`` si elle existait. """
126 newquote
= Quote(self
._collapse
_author
(author
), content
, timestamp
)
127 if not newquote
in self
.quotelist
:
129 self
.quotelist
.append(newquote
)
134 return repr(self
.quotelist
)
137 """ Sort une quote aléatoire """
138 return random
.choice(self
.quotelist
)
139 def quotesfrom(self
, author
):
140 """ Sort toutes les quotes de ``author`` """
141 return [q
for q
in self
.quotelist
if q
.author
== author
]
142 def randomfrom(self
, author
):
143 """ Sort une quote aléatoire de ``author`` """
144 return random
.choice(self
.quotesfrom(author
))
146 def search(self
, inquote
=None, author
=None, regexp
=False):
147 """Fait une recherche dans les quotes."""
153 qreg
= re
.compile(inquote
, flags
=re
.UNICODE
)
154 areg
= re
.compile(author
, flags
=re
.UNICODE
)
155 l
= [q
for q
in self
.quotelist
if qreg
.match(q
.content
) and areg
.match(q
.author
)]
161 l
= [q
for q
in self
.quotelist
if inquote
in q
.content
and author
in q
.author
]
164 def search_authors(self
, author
=None, regexp
=False):
165 """Renvoie la liste des auteurs contenant ``author`` ou qui matchent la regexp."""
169 areg
= re
.compile(author
, flags
=re
.UNICODE
)
170 l
= list(set([q
.author
for q
in self
.quotelist
if areg
.match(q
.author
)]))
174 l
= list(set([q
.author
for q
in self
.quotelist
if author
in q
.author
]))
177 def dump(quotedb
, dump_file
=None):
178 """Pour exporter les quotes dans un format readable vers un fichier."""
179 if dump_file
is None:
180 dump_file
= config
.quote_dump_file
181 t
= "\n".join(["%s %s" % (q
.timestamp
.strftime("%F_%T"), q
) for q
in quotedb
.quotelist
]) + "\n"
182 with
open(dump_file
, "w") as f
:
185 def restore(dump_file
=None):
186 """Crée un DB de quotes en parsant le contenu d'un fichier de dump."""
187 if dump_file
is None:
188 dump_file
= config
.quote_dump_file
189 with
open(dump_file
) as f
:
191 t
= t
.decode("utf-8") # Oui, ça peut fail, mais on ne doit alors pas continuer
192 l
= [m
.groupdict() for m
in quote_matcher_with_timestamp
.finditer(t
)]
193 # On instancie les quotes grâce aux dicos qui ont déjà la bonne tronche
194 l
= [Quote(**q
) for q
in l
]
195 newquotedb
= QuoteDB()
196 newquotedb
.quotelist
= l