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