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, place
=None, quoter
=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
55 d
= {"author" : self
.author
, "content" : self
.content
,
56 "timestamp" : self
.timestamp
.strftime(u
"%F_%T"),
57 "place" : self
.place
, "quoter" : self
.quoter
}
60 def __unicode__(self
):
61 """ Retourne la quote affichable """
62 return config
.quote_template
% self
.__dict
__
64 return unicode(self
).encode("utf-8")
67 """ Retourne une chaîne représentant la totalité des infos de la quote,
68 tout en étant parsable et human-readable. """
69 place
= self
.place
if self
.place
else ""
70 quoter
= self
.quoter
if self
.quoter
else ""
71 return (u
"%s %s | %s | %s" % (self
.timestamp
.strftime("%F_%T"), unicode(self
), place
, quoter
)).encode("utf-8")
73 def __eq__(self
, otherquote
):
74 """ Vérifie si cette phrase n'a pas déjà été dite par la même personne.
75 Indépendamment de la date et de la casse. """
76 return [self
.author
.lower(), self
.content
.lower()] == [otherquote
.author
.lower(), otherquote
.content
.lower()]
79 def parse(text
, date
=None):
80 """ Parse le ``text`` et renvoie une quote ou None. """
83 get
= quote_matcher
.match(text
)
86 return Quote(d
["author"], d
["content"], date
)
88 def load_file(filename
):
89 """ Récupère les quotes depuis le fichier """
90 with
open(filename
) as f
:
91 jsonquotes
= json
.load(f
)
92 quotes
= [Quote(**q
) for q
in jsonquotes
]
95 def save_file(quotes
, filename
):
96 """ Enregistre les quotes dans le fichier """
97 with
open(filename
, "w") as f
:
98 raws
= [q
.jsonize() for q
in quotes
]
101 class QuoteDB(object):
102 """ Stocke et distribue des quotes. """
107 """ Charge le fichier de quotes dans la DB """
108 self
.quotelist
= load_file(config
.quote_file
)
111 """ Sauvegarde la DB dans le fichier de quotes """
112 save_file(self
.quotelist
, config
.quote_file
)
114 def _collapse_author(self
, author
):
115 """ Renvoie ``author`` avec la casse déjà utilisée si il a déjà été quoté
116 sinon, le renvoie sans le modifier. """
117 authors
= list(set([q
.author
for q
in self
.quotelist
if q
.author
.lower() == author
.lower()]))
119 print "Warning : authors %s" % authors
125 def get_clash_authors(self
):
126 """ Renvoie une liste de liste d'auteurs qui sont enregistrés avec des casses différentes. """
127 authors
= list(set([q
.author
for q
in self
.quotelist
]))
128 authors
= equivalence_partition(authors
, lambda x
,y
: x
.lower() == y
.lower())
129 authors
= [list(c
) for c
in authors
if len(c
) > 1]
132 def store(self
, timestamp
=None, **kwargs
):
133 """ Enregistre une nouvelle quote, sauf si elle existe déjà.
134 Force l'auteur à utiliser la même casse si un auteur de casse différente existait déjà.
135 Renvoie ``True`` si elle a été ajoutée, ``False`` si elle existait. """
136 kwargs
["author"] = self
._collapse
_author
(kwargs
["author"])
137 kwargs
["timestamp"] = timestamp
138 newquote
= Quote(**kwargs
)
139 if not newquote
in self
.quotelist
:
141 self
.quotelist
.append(newquote
)
146 return repr(self
.quotelist
)
149 """ Sort une quote aléatoire """
150 return random
.choice(self
.quotelist
)
152 def quotesfrom(self
, author
):
153 """ Sort toutes les quotes de ``author`` """
154 return [q
for q
in self
.quotelist
if q
.author
== author
]
155 def randomfrom(self
, author
):
156 """ Sort une quote aléatoire de ``author`` """
157 return random
.choice(self
.quotesfrom(author
))
159 def search(self
, inquote
=None, author
=None, regexp
=False):
160 """Fait une recherche dans les quotes."""
166 qreg
= re
.compile(inquote
, flags
=re
.UNICODE
)
167 areg
= re
.compile(author
, flags
=re
.UNICODE
)
168 l
= [q
for q
in self
.quotelist
if qreg
.match(q
.content
) and areg
.match(q
.author
)]
174 l
= [q
for q
in self
.quotelist
if inquote
in q
.content
and author
in q
.author
]
177 def search_authors(self
, author
=None, regexp
=False):
178 """Renvoie la liste des auteurs contenant ``author`` ou qui matchent la regexp."""
182 areg
= re
.compile(author
, flags
=re
.UNICODE
)
183 l
= list(set([q
.author
for q
in self
.quotelist
if areg
.match(q
.author
)]))
187 l
= list(set([q
.author
for q
in self
.quotelist
if author
in q
.author
]))
190 def dump(quotedb
, dump_file
=None):
191 """Pour exporter les quotes dans un format readable vers un fichier."""
192 if dump_file
is None:
193 dump_file
= config
.quote_dump_file
194 t
= "\n".join([q
.full_str() for q
in quotedb
.quotelist
]) + "\n"
195 with
open(dump_file
, "w") as f
:
198 def restore(dump_file
=None):
199 """Crée un DB de quotes en parsant le contenu d'un fichier de dump."""
200 if dump_file
is None:
201 dump_file
= config
.quote_dump_file
202 with
open(dump_file
) as f
:
204 t
= t
.decode("utf-8") # Oui, ça peut fail, mais on ne doit alors pas continuer
205 l
= [m
.groupdict() for m
in quote_matcher_with_timestamp
.finditer(t
)]
206 # On instancie les quotes grâce aux dicos qui ont déjà la bonne tronche
207 l
= [Quote(**q
) for q
in l
]
208 newquotedb
= QuoteDB()
209 newquotedb
.quotelist
= l