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