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