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