Nejlepsi posun integrovan do robota
[krypto.git] / robot.py
1 #!/usr/bin/env python
2
3 import sys
4 import smtplib
5 import email
6 from email.mime.text import MIMEText
7 from time import asctime
8 from ocesavac import ocesat
9 import jakobsen
10 from jakobsen import reference
11 import koincidence
12 import posuny
13 import pickle
14 import os
15
16 ABECEDA = 'ABCDEFGHIJKLMNOPQRSTUVWXYZ'
17
18 class settings:
19     pass
20
21 settings = pickle.load(open('./robot_settings'))
22 os.chdir(settings.path)
23
24 def posli_mail(komu, co, text):
25     msg = MIMEText(text)
26     msg['Subject'] = co
27     msg['From'] = settings.name + " <" + settings.mail  + ">"
28     msg['To'] = komu
29     smtplib.SMTP('localhost').sendmail(settings.mail, komu, msg.as_string())
30
31 # Cteni mailu
32 mail = email.message_from_file(sys.stdin)
33 predmet, _, opsny = mail['Subject'].strip().upper().partition(' ')
34 odesilatel = mail['From']
35 if predmet != settings.subject:
36     sys.exit()
37 if opsny == 'H':
38     posli_mail(odesilatel, "Napoveda", "Napoveda:\n"
39                "Prikazy robotovi vkladejte do predmetu zpravy zaslane na adresu " + 
40                settings.mail + ".\nNa velikosti pismen nezalezi."
41                "\n\nSeznam prikazu:\n"
42                "\"" + settings.subject + "\"   Analyza zpravy z tela mailu.\n" 
43                "\"" + settings.subject + " L\" Opakovana analyza posledni zpravy.\n" +
44                "\"" + settings.subject + " X\" Vynuti analyzu bez mezer.\n"
45                "\"" + settings.subject + " M\" Vynuti analyzu s mezerami.\n"
46                "\"" + settings.subject + " B\" Vypise cetnost vsech bigramu.\n"
47                "\"" + settings.subject + " P\" Vypise vsechny posunuti textu.\n"
48                "\"" + settings.subject + " A\" Vypise uplne vsechno co dava "
49                "alespon trochu smysl (nedoporucuji).\n"
50                "\"" + settings.subject + " H\" Tato napoveda.\n"
51                "\nVolby (mimo \"H\") je mozno libovolne kombinovat.\n"
52                "Napriklad \"" + settings.subject + " LXB\" analyzuje posledni "
53                "prijatou zpravu s vynucenim analyzy bez mezer a navic vypise "
54                "cetnosti bigramu.\n"
55                "\nS pozdravem\nVas Robot\n")
56     sys.exit()
57 if 'L' in opsny:
58     mail = pickle.load(open('./msg_last.tmp'))
59     if (mail['From'] != odesilatel):
60         posli_mail(odesilatel, "Omluva", "Omlouvam se,"
61                    " ale Vase zprava uz bohuzel neni na serveru ulozena.\n")
62         sys.exit()
63 telo = mail.get_payload().strip()
64 if len(telo) < 2:
65     sys.exit()
66 zprava = telo.upper()
67 analyza = []
68
69 # Puvodni zprava
70 analyza.append("Puvodni zprava:")
71 analyza.append(telo)
72
73 # Prepinani spacemodu
74 mod_m = False
75 mod_x = False
76 if 'X' in opsny:
77     mod_x = True
78 if 'M' in opsny:
79     mod_m = True
80 if 'A' in opsny:
81     mod_x = True
82     mod_m = True
83 if not mod_m and not mod_x:
84     if ' ' in ocesat(zprava, True):
85         mod_m = True
86         analyza.append("\nAutomaticky zvolena analyza s mezerami.")
87     else:
88         mod_x = True
89         analyza.append("\nAutomaticky zvolena analyza bez mezer.")
90     
91 # Zakladni udaje
92 analyza.append("\nZakladni udaje:")
93 if mod_m:
94     analyza.append("Delka zpravy (vcetne mezer a zvlastnich znaku): " + str(len(zprava)))
95 analyza.append("Delka zpravy (bez mezer a zvlastnich znaku): " + str(len(ocesat(zprava, False))))
96
97 # Prumerna delka slova
98 if mod_m:
99     analyza.append("\nPrumerna delka slova:")
100     slova = ocesat(zprava, True).split()
101     analyza.append('{0:.3}'.format(float(sum([len(word) for word in slova]))/len(slova)))
102
103 # Frekvence znaku 
104 analyza.append("\nFrekvencni analyza:")
105 freq = dict()
106 for char in ABECEDA:
107     freq[char] = 0
108 for char in zprava:
109     if char in freq:
110         freq[char] += 1
111 celkem = sum(freq.values())
112 for char in sorted(freq, key=freq.get, reverse=True):
113     analyza.append('{0} {1:>3} {2:>6.2%}'.format(char, freq[char],
114                    float(freq[char])/celkem))
115
116 # Index koincidence
117 # TODO verze s mezerami
118 analyza.append("\nIndexy koincidence (vzdy bez mezer):")
119 analyza.append(koincidence.tabulka_indexu(zprava))
120
121 # Jakobsen
122 if mod_m:
123     analyza.append("\nJakobsen (s mezerami):")
124     analyza.append(jakobsen.desifruj(zprava, './ref'))
125 if mod_x:
126     analyza.append("\nJakobsen (bez mezer):")
127     analyza.append(jakobsen.desifruj(zprava, './bref'))
128
129 # Posuny v abecede
130 (posun, posunuty_text) = posuny.nejlepsi(zprava)
131 analyza.append("\nNejlepsi posun ({0}):".format(posun))
132 analyza.append(posunuty_text)
133
134 if 'P' in opsny or 'A' in opsny:
135     analyza.append("\nPosuny v abecede:")
136     analyza.append(posuny.vsechny(zprava))
137
138 # Bigramy
139 if ('B' in opsny or 'A' in opsny):
140     analyza.append("\nBigramy podle cetnosti:")
141     bigramy = dict()
142     for i in ABECEDA:
143         for j in ABECEDA:
144             bigramy[i+j] = 0
145     for i in range(0, len(zprava)-1):
146         if zprava[i:i+2] in bigramy:
147             bigramy[zprava[i:i+2]] += 1
148     celkem = sum(bigramy.values())
149     for char in sorted(bigramy, key=bigramy.get, reverse=True):
150         if bigramy[char] == 0:
151             break
152         analyza.append('{0} {1:>3} {2:>6.2%}'.format(char, bigramy[char],
153                        float(bigramy[char])/celkem))
154
155 # Podpis a pozdrav
156 analyza.append(''.join(["\nS pozdravem\nVas Robot\n"
157                         "\n--\nPro napovedu zaslete mail s predmetem \"",
158                         settings.subject," H\".\n"]))
159
160 # Kompozice odpovedi
161 msg = MIMEText("\n".join(analyza))
162 msg['Subject'] = "Analyza Vasi zpravy"
163 msg['From'] = settings.name + " <" + settings.mail  + ">"
164 msg['To'] = odesilatel
165
166 # Odeslani odpovedi
167 s = smtplib.SMTP('localhost')
168 s.sendmail(settings.mail, odesilatel, msg.as_string())
169 s.quit()
170
171 # Zalogovani posledni zpravy
172 f = open('./robot_last.tmp', 'w')
173 f.write(msg.as_string())
174 f.close()
175
176 f = open('./msg_last.tmp', 'w')
177 pickle.dump(mail, f)
178 f.close()
179
180 f = open('./robot.log', 'a')
181 f.write(asctime() + " " +  odesilatel + "\n")
182 f.close()