[Author Prev][Author Next][Thread Prev][Thread Next][Author Index][Thread Index]

[or-cvs] r18490: {torflow} Deprecate soatcli. Improve snakeinspeector for html diffing. (torflow/trunk/NetworkScanners)



Author: mikeperry
Date: 2009-02-11 10:23:39 -0500 (Wed, 11 Feb 2009)
New Revision: 18490

Removed:
   torflow/trunk/NetworkScanners/soatcli.py
Modified:
   torflow/trunk/NetworkScanners/libsoat.py
   torflow/trunk/NetworkScanners/snakeinspector.py
   torflow/trunk/NetworkScanners/soat.py
Log:

Deprecate soatcli. Improve snakeinspeector for html diffing.
Add unique filename enforcement.



Modified: torflow/trunk/NetworkScanners/libsoat.py
===================================================================
--- torflow/trunk/NetworkScanners/libsoat.py	2009-02-11 13:48:09 UTC (rev 18489)
+++ torflow/trunk/NetworkScanners/libsoat.py	2009-02-11 15:23:39 UTC (rev 18490)
@@ -74,7 +74,8 @@
     self.reason = reason
     self.false_positive=False
     self.false_positive_reason="None"
-    self.verbose=False
+    self.verbose=0
+    self.filename=None
 
   def _rebase(self, filename, new_data_root):
     if not filename: return filename
@@ -83,7 +84,8 @@
     return os.path.normpath(os.path.join(new_data_root, *split_file[1:]))
 
   def rebase(self, new_data_root):
-    pass
+    if 'filename' in self.__dict__: # XXX: Kill this...
+      self.filename = self._rebase(self.filename, new_data_root)
  
   def mark_false_positive(self, reason):
     self.false_positive=True
@@ -147,15 +149,14 @@
     if self.verbose:
       for cert in ssl_domain.cert_map.iterkeys():
         ret += "\nCert for "+ssl_domain.cert_map[cert]+":\n"
-        ret += cert
+        if self.verbose > 1: ret += cert
         ret += self._dump_cert(cert)
       if self.exit_cert:
-        # XXX: Kill the first part of this clause after restart:
-        if 'exit_ip' in self.__dict__ and self.exit_ip: 
+        if self.exit_ip: 
           ret += "\nExit node's cert for "+self.exit_ip+":\n"
         else:
           ret += "\nExit node's cert:\n"
-        ret += self.exit_cert
+        if self.verbose > 1: ret += self.exit_cert
         ret += self._dump_cert(self.exit_cert)
     return ret 
 
@@ -332,29 +333,35 @@
     ret = TestResult.__str__(self)
     if self.verbose:
       soup = old_soup = tor_soup = None
-      if self.content and self.content_old:
+      if self.content:
         content = open(self.content).read().decode('ascii', 'ignore')
+        soup = FullyStrainedSoup(content)
+
+      if self.content_old:
         content_old = open(self.content_old).read().decode('ascii', 'ignore')
-        soup = FullyStrainedSoup(content)
         old_soup = FullyStrainedSoup(content_old)
-        tags = map(str, soup.findAll())
-        old_tags = map(str, old_soup.findAll())
-        diff = difflib.unified_diff(old_tags, tags, "Non-Tor1", "Non-Tor2",
-                                    lineterm="")
-        for line in diff:
-          ret+=line+"\n"
-      if self.content and self.content_exit:
-        content = open(self.content).read().decode('ascii', 'ignore')
+
+      if self.content_exit:
         content_exit = open(self.content_exit).read().decode('ascii', 'ignore')
-        soup = FullyStrainedSoup(content)
         tor_soup = FullyStrainedSoup(content_exit)
-        tags = map(str, soup.findAll())
-        tor_tags = map(str, tor_soup.findAll())
-        diff = difflib.unified_diff(tags, tor_tags, "Non-Tor", "Exit",
-                                    lineterm="")
-        for line in diff:
-          ret+=line+"\n"
 
+      if self.verbose > 1:
+        if self.content and self.content_old:
+          tags = map(str, soup.findAll())
+          old_tags = map(str, old_soup.findAll())
+          diff = difflib.unified_diff(old_tags, tags, "Non-Tor1", "Non-Tor2",
+                                      lineterm="")
+          for line in diff:
+            ret+=line+"\n"
+
+        if self.content and self.content_exit:
+          tags = map(str, soup.findAll())
+          tor_tags = map(str, tor_soup.findAll())
+          diff = difflib.unified_diff(tags, tor_tags, "Non-Tor", "Exit",
+                                      lineterm="")
+          for line in diff:
+            ret+=line+"\n"
+
       if soup and tor_soup and old_soup:
         old_vs_new = SoupDiffer(old_soup, soup)
         new_vs_old = SoupDiffer(soup, old_soup)
@@ -369,15 +376,28 @@
                                 old_vs_new.changed_attributes_by_tag(),
                                 new_vs_old.changed_attributes_by_tag())
 
-        changed_content = bool(old_vs_new.changed_content() or old_vs_new.changed_content())
-     
-        ret += "\nTor changed tags:\n"
-        ret += new_vs_tor.more_changed_tags(changed_tags)
-        ret += "\nTor changed attrs:\n"
-        ret += new_vs_tor.more_changed_attrs(changed_attributes)
-        if not changed_content:
+        changed_content = bool(new_vs_old.changed_content() or old_vs_new.changed_content())
+
+        more_tags = new_vs_tor.more_changed_tags(changed_tags)     
+        more_attrs = new_vs_tor.more_changed_attrs(changed_attributes)
+        more_content = new_vs_tor.changed_content()
+
+        if more_tags:
+          ret += "\nTor changed tags:\n"
+          ret += more_tags
+        if more_attrs:
+          ret += "\nTor changed attrs:\n"
+          ret += more_attrs
+        if not changed_content and more_content:
           ret += "\nChanged Content:\n"
-          ret += "\n".join(new_vs_tor.changed_content())+"\n"
+          ret += "\n".join(more_content)+"\n"
+        if (changed_content or not more_content) and not more_tags and not more_attrs:
+          ret += "\nSoupDiffer claims false positive.\n"
+          jsdiff = JSSoupDiffer(old_soup)
+          jsdiff.prune_differences(soup)
+          jsdifferences = jsdiff.show_differences(tor_soup)
+          if not jsdifferences: jsdifferences = "None."
+          ret += "Javascript Differences: "+jsdifferences+"\n"
     else:
       if self.content:
         ret += " "+self.content+"\n"
@@ -512,15 +532,17 @@
     fh = open(file, 'r')
     return pickle.load(fh)
 
-  def uniqueFilename(self, afile):
+  def uniqueFilename(afile):
     if not os.path.exists(afile):
       return afile
+    (prefix,suffix)=os.path.splitext(afile)
     i=1
-    while os.path.exists(afile+"."+str(i)):
+    while os.path.exists(prefix+"."+str(i)+suffix):
       i+=1
-    return afile+"."+str(i) 
+    return prefix+"."+str(i)+suffix
+  uniqueFilename = Callable(uniqueFilename)
   
-  def safeFilename(self, unsafe_file):
+  def safeFilename(unsafe_file):
     ''' 
     remove characters illegal in some systems 
     and trim the string to a reasonable length
@@ -528,16 +550,16 @@
     unsafe_file = unsafe_file.decode('ascii', 'ignore')
     safe_file = re.sub(unsafe_filechars, "_", unsafe_file)
     return str(safe_file[:200])
+  safeFilename = Callable(safeFilename)
 
-  def resultFilename(self, result):
-    # XXX: Check existance and make a secondary name if exists.
+  def __resultFilename(self, result):
     address = ''
     if result.__class__.__name__ == 'HtmlTestResult' or result.__class__.__name__ == 'HttpTestResult':
-      address = self.safeFilename(result.site[7:])
+      address = DataHandler.safeFilename(result.site[7:])
     elif result.__class__.__name__ == 'SSLTestResult':
-      address = self.safeFilename(result.site[8:])
+      address = DataHandler.safeFilename(result.site[8:])
     elif 'TestResult' in result.__class__.__name__:
-      address = self.safeFilename(result.site)
+      address = DataHandler.safeFilename(result.site)
     else:
       raise Exception, 'This doesn\'t seems to be a result instance.'
 
@@ -551,11 +573,12 @@
     elif result.status == TEST_FAILURE:
       rdir += 'failed/'
 
-    return str((rdir+address+'.'+result.exit_node[1:]+".result").decode('ascii', 'ignore'))
+    return DataHandler.uniqueFilename(str((rdir+address+'.'+result.exit_node[1:]+".result").decode('ascii', 'ignore')))
 
   def saveResult(self, result):
     ''' generic method for saving test results '''
-    result_file = open(self.resultFilename(result), 'w')
+    result.filename = self.__resultFilename(result)
+    result_file = open(result.filename, 'w')
     pickle.dump(result, result_file)
     result_file.close()
 
@@ -805,6 +828,31 @@
         return True
     return False
 
+  def _difference_printer(self, other_cnts):
+    ret = ""
+    missing = []
+    miscount = []
+    new = []
+    for node in self.ast_cnts.iterkeys():
+      if not self.ast_cnts[node]: continue # pruned difference
+      if node not in other_cnts:
+        missing.append(str(node))
+      elif self.ast_cnts[node] != other_cnts[node]:
+        miscount.append(str(node))
+    for node in other_cnts.iterkeys():
+      if node not in self.ast_cnts:
+        new.append(str(node))
+    if missing:
+      ret += "\nMissing: "
+      for node in missing: ret += node
+    if new:
+      ret += "\nNew: "
+      for node in new: ret += node
+    if miscount:
+      ret += "\nMiscount: "
+      for node in miscount: ret += node
+    return ret
+
   def prune_differences(self, other_string):
     if not HAVE_PYPY: return
     other_cnts = self._count_ast_elements(other_string)
@@ -817,6 +865,14 @@
     other_cnts = self._count_ast_elements(other_string)
     return self._difference_checker(other_cnts) 
 
+  def show_differences(self, other_string):
+    ret = ""
+    if not HAVE_PYPY:
+      return "PyPy import not present. Not diffing javascript"
+    other_cnts = self._count_ast_elements(other_string)
+    return self._difference_printer(other_cnts) 
+
+
 class JSSoupDiffer(JSDiffer):
   def _add_cnts(tag_cnts, ast_cnts):
     ret_cnts = {}
@@ -855,14 +911,4 @@
         ast_cnts = JSSoupDiffer._add_cnts(tag_cnts, ast_cnts)
     return ast_cnts
 
-  def prune_differences(self, other_soup):
-    if not HAVE_PYPY: return
-    other_cnts = self._count_ast_elements(other_soup)
-    self._difference_pruner(other_cnts)
 
-  def contains_differences(self, other_soup):
-    if not HAVE_PYPY:
-      plog("NOTICE", "PyPy import not present. Not diffing javascript")
-      return False
-    other_cnts = self._count_ast_elements(other_soup)
-    return self._difference_checker(other_cnts) 

Modified: torflow/trunk/NetworkScanners/snakeinspector.py
===================================================================
--- torflow/trunk/NetworkScanners/snakeinspector.py	2009-02-11 13:48:09 UTC (rev 18489)
+++ torflow/trunk/NetworkScanners/snakeinspector.py	2009-02-11 15:23:39 UTC (rev 18490)
@@ -36,7 +36,7 @@
   node=None
   reason=None
   result=None
-  verbose=False
+  verbose=0
   proto=None
   resultfilter=None
   for o,a in opts:
@@ -49,7 +49,7 @@
     elif o == '-r' or o == '--reason': 
       reason = a
     elif o == '-v' or o == '--verbose': 
-      verbose = True
+      verbose += 1
     elif o == '-t' or o == '--resultfilter':
       resultfilter = a
     elif o == '-p' or o == '--proto':

Modified: torflow/trunk/NetworkScanners/soat.py
===================================================================
--- torflow/trunk/NetworkScanners/soat.py	2009-02-11 13:48:09 UTC (rev 18489)
+++ torflow/trunk/NetworkScanners/soat.py	2009-02-11 15:23:39 UTC (rev 18490)
@@ -207,7 +207,7 @@
         # Save this new result file in false positive dir 
         # and remove old one
         try:
-          os.unlink(self.datahandler.resultFilename(r))
+          os.unlink(r.filename)
         except:
           pass
         r.mark_false_positive(reason)
@@ -513,7 +513,7 @@
     ''' check whether a http connection to a given address is molested '''
 
     # an address representation acceptable for a filename 
-    address_file = self.datahandler.safeFilename(address[7:])
+    address_file = DataHandler.safeFilename(address[7:])
     content_prefix = http_content_dir+address_file
     
     # Keep a copy of the cookie jar before mods for refetch or
@@ -730,15 +730,14 @@
   def _check_http_worker(self, address, http_ret):
     (mime_type,pcontent,psha1sum,content,sha1sum,content_new,sha1sum_new,exit_node) = http_ret
      
-    address_file = self.datahandler.safeFilename(address[7:])
+    address_file = DataHandler.safeFilename(address[7:])
     content_prefix = http_content_dir+address_file
     failed_prefix = http_failed_dir+address_file
 
     # compare the new and old content
     # if they match, means the node has been changing the content
     if sha1sum.hexdigest() == sha1sum_new.hexdigest():
-      # XXX: Check for existence of this file before overwriting
-      exit_content_file = open(failed_prefix+'.'+exit_node[1:]+'.content', 'w')
+      exit_content_file = open(DataHandler.uniqueFilename(failed_prefix+'.'+exit_node[1:]+'.content'), 'w')
       exit_content_file.write(pcontent)
       exit_content_file.close()
 
@@ -752,8 +751,7 @@
       self.register_exit_failure(address, exit_node)
       return TEST_FAILURE
 
-    # XXX: Check for existence of this file before overwriting
-    exit_content_file = open(failed_prefix+'.'+exit_node[1:]+'.dyn-content','w')
+    exit_content_file = open(DataHandler.uniqueFilename(failed_prefix+'.'+exit_node[1:]+'.dyn-content'),'w')
     exit_content_file.write(pcontent)
     exit_content_file.close()
 
@@ -891,12 +889,11 @@
       else: self.successes[address]=1
       return TEST_SUCCESS
     else:
-      address_file = self.datahandler.safeFilename(address[7:])
+      address_file = DataHandler.safeFilename(address[7:])
       content_prefix = http_content_dir+address_file
       failed_prefix = http_failed_dir+address_file
 
-      # XXX: Check for existence of this file before overwriting
-      exit_content_file = open(failed_prefix+'.'+exit_node[1:]+'.dyn-content',
+      exit_content_file = open(DataHandler.uniqueFilename(failed_prefix+'.'+exit_node[1:]+'.dyn-content'),
                                 'w')
       exit_content_file.write(tor_js)
       exit_content_file.close()
@@ -932,7 +929,7 @@
         return self._check_http_worker(address, http_ret)
 
     # an address representation acceptable for a filename 
-    address_file = self.datahandler.safeFilename(address[7:])
+    address_file = DataHandler.safeFilename(address[7:])
     content_prefix = http_content_dir+address_file
     failed_prefix = http_failed_dir+address_file
 
@@ -971,8 +968,7 @@
     # compare the new and old content
     # if they match, means the node has been changing the content
     if str(orig_soup) == str(new_soup):
-      # XXX: Check for existence of this file before overwriting
-      exit_content_file = open(failed_prefix+'.'+exit_node[1:]+'.content', 'w')
+      exit_content_file = open(DataHandler.uniqueFilename(failed_prefix+'.'+exit_node[1:]+'.content'), 'w')
       exit_content_file.write(tor_html)
       exit_content_file.close()
 
@@ -1004,7 +1000,7 @@
                             old_vs_new.changed_attributes_by_tag(),
                             new_vs_old.changed_attributes_by_tag())
 
-    changed_content = bool(old_vs_new.changed_content() or old_vs_new.changed_content())
+    changed_content = bool(new_vs_old.changed_content() or old_vs_new.changed_content())
  
     # Verify all of our changed tags are present here 
     if new_vs_tor.has_more_changed_tags(changed_tags) or \
@@ -1013,11 +1009,14 @@
       false_positive = False
     else:
       false_positive = True
+      
+    plog("INFO", "SoupDiffer predicts false_positive="+str(false_positive))
 
     if false_positive:
       jsdiff = JSSoupDiffer(orig_soup)
       jsdiff.prune_differences(new_soup)
       false_positive = not jsdiff.contains_differences(tor_soup)
+      plog("INFO", "JSSoupDiffer predicts false_positive="+str(false_positive))
 
     if false_positive:
       plog("NOTICE", "False positive detected for dynamic change at "+address+" via "+exit_node)
@@ -1028,8 +1027,7 @@
       else: self.successes[address]=1
       return TEST_SUCCESS
 
-    # XXX: Check for existence of this file before overwriting
-    exit_content_file = open(failed_prefix+'.'+exit_node[1:]+'.dyn-content','w')
+    exit_content_file = open(DataHandler.uniqueFilename(failed_prefix+'.'+exit_node[1:]+'.dyn-content'),'w')
     exit_content_file.write(tor_html)
     exit_content_file.close()
 
@@ -1124,7 +1122,7 @@
     plog('INFO', 'Conducting an ssl test with destination ' + address)
 
     # an address representation acceptable for a filename 
-    address_file = self.datahandler.safeFilename(address[8:])
+    address_file = DataHandler.safeFilename(address[8:])
     ssl_file_name = ssl_certs_dir + address_file + '.ssl'
 
     # load the original cert and compare

Deleted: torflow/trunk/NetworkScanners/soatcli.py
===================================================================
--- torflow/trunk/NetworkScanners/soatcli.py	2009-02-11 13:48:09 UTC (rev 18489)
+++ torflow/trunk/NetworkScanners/soatcli.py	2009-02-11 15:23:39 UTC (rev 18490)
@@ -1,418 +0,0 @@
-#!/usr/bin/python
-#
-# 2008 Aleksei Gorny, mentored by Mike Perry
-
-import dircache
-import operator
-import os
-import pickle
-import sys
-import time
-
-import sets
-from sets import Set
-
-import libsoat
-from libsoat import *
-
-#
-# Displaying stats on the console
-#
-
-class StatsConsole:
-  ''' Class to display statistics from CLI'''
-  
-  def Listen(self):
-    while 1:
-      input = raw_input(">>>")
-      if input == 'e' or input == 'exit':
-        exit()
-      elif input == 's' or input == 'summary':
-        self.Summary()
-      elif input == 'h' or input == 'help' or len(input) > 6:
-        self.Help() 
-      else:
-        self.Reply(input)
-  
-  def Summary(self):
-    dh = DataHandler()
-    data = dh.getAll()
-    
-    nodeSet = Set([])
-    sshSet = Set([])
-    sslSet = Set([])
-    httpSet = Set([])
-    smtpSet = Set([])
-    popSet = Set([])
-    imapSet = Set([])
-    dnsSet = Set([])
-    dnsrebindSet = Set([])
-
-    total = len(data)
-    good = bad = inconclusive = 0
-    ssh = http = ssl = pop = imap = smtp = dns = dnsrebind = 0
-
-    for result in data:
-      nodeSet.add(result.exit_node)
-      
-      if result.status == 0:
-        good += 1
-      elif result.status == 1:
-        inconclusive += 1
-      elif result.status == 2:
-        bad += 1
-      
-      if result.__class__.__name__ == 'SSHTestResult':
-        sshSet.add(result.exit_node)
-        ssh += 1
-      elif result.__class__.__name__ == 'HttpTestResult' or result.__class__.__name__ == 'HtmlTestResult':
-        httpSet.add(result.exit_node)
-        http += 1
-      elif result.__class__.__name__ == 'SSLTestResult':
-        sslSet.add(result.exit_node)
-        ssl += 1
-      elif result.__class__.__name__ == 'IMAPTestResult':
-        imapSet.add(result.exit_node)
-        imap += 1
-      elif result.__class__.__name__ == 'POPTestResult':
-        popSet.add(result.exit_node)
-        pop += 1
-      elif result.__class__.__name__ == 'SMTPTestResult':
-        smtpSet.add(result.exit_node)
-        smtp += 1
-      elif result.__class__.__name__ == 'DNSTestResult':
-        dnsSet.add(result.exit_node)
-        dns += 1
-      elif result.__class__.__name__ == 'DNSRebindTestResult':
-        dnsrebindSet.add(result.exit_node)
-        dnsrebind += 1
-
-    swidth = 25
-    nwidth = 10
-    width = swidth + nwidth
-
-    header_format = '%-*s%*s'
-    format = '%-*s%*i'
-
-    print '=' * width
-    print header_format % (swidth, 'Parameter', nwidth, 'Count')
-    print '-' * width
-
-    stats = [
-      ('Tests completed', total),
-      ('Nodes tested', len(nodeSet)),
-      ('Nodes SSL-tested', len(sslSet)),
-      ('Nodes HTTP-tested', len(httpSet)),
-      ('Nodes SSH-tested', len(sshSet)),
-      ('Nodes POP-tested', len(popSet)),
-      ('Nodes IMAP-tested', len(imapSet)),
-      ('Nodes SMTP-tested', len(smtpSet)),
-      ('Nodes DNS-tested', len(dnsSet)),
-      ('Nodes DNSRebind-tested', len(dnsrebindSet)),
-      ('Failed tests', bad),
-      ('Succeeded tests', good),
-      ('Inconclusive tests', inconclusive),
-      ('SSH tests', ssh),
-      ('HTTP tests', http),
-      ('SSL tests', ssl),
-      ('POP tests', pop),
-      ('IMAP tests', imap),
-      ('SMTP tests', smtp),
-      ('DNS tests', dns),
-      ('DNS rebind tests', dnsrebind)
-    ]
-
-    for (k,v) in stats:
-      print format % (swidth, k, nwidth, v)
-    print '=' * width
-
-  def Reply(self, input):
-
-    good = bad = inconclusive = False
-    protocols = []
-
-    if 'a' in input:
-      good = bad = inconclusive = True
-      protocols.extend(["ssh", "http", "ssl", "imap", "pop", "smtp"])
-    else:
-      good = 'g' in input
-      bad = 'b' in input
-      inconclusive = 'i' in input
-
-      if 's' in input:
-        protocols.append("ssh")
-      if 'h' in input:
-        protocols.append("http")
-      if 'l' in input:
-        protocols.append("ssl")
-      if 'p' in input:
-        protocols.append("imap")
-      if 'o' in input:
-        protocols.append("pop")
-      if 't' in input:
-        protocols.append("smtp")
-      if 'd' in input:
-        protocols.append("dns")
-      if 'r' in input:
-        protocols.append("dnsrebind")
-
-    dh = DataHandler()
-    data = dh.getAll()
-    filtered = dh.filterResults(data, protocols, good, bad, inconclusive)
-
-    nodewidth = 45
-    typewidth = 10
-    sitewidth = 30
-    timewidth = 30
-    statuswidth = 6
-    width = nodewidth + typewidth + sitewidth + timewidth + statuswidth
-
-    format = '%-*s%-*s%-*s%-*s%-*s'
-
-    print '=' * width 
-    print format % (nodewidth, 'Exit node', typewidth, 'Test type', sitewidth, 'Remote site', 
-        timewidth, 'Time', statuswidth, 'Status')
-    print '-' * width
-    for result in filtered:
-      print format % (nodewidth, `result.exit_node`, 
-          typewidth, result.__class__.__name__[:-10],
-          sitewidth, result.site, 
-          timewidth, time.strftime("%a, %d %b %Y %H:%M:%S", time.localtime(result.timestamp)), 
-          statuswidth, `result.status`)
-    print '=' * width
-
-  def Help(self):
-    print ''
-    print 'Options:'
-    print '* summmary (s) - display a short summary about all tests done so far'
-    print '* exit (e) - terminate the program'
-    print '* help (h) - display this help text'
-    print '* all (a) - list all the results'
-    print '* (shlgbi) - display a filtered list of test results. Letters are optional and mean the following:'
-    print '  s - show ssh results'
-    print '  h - show http results'
-    print '  l - show ssl results'
-    print '  g - show good results'
-    print '  b - show bad results'
-    print '  i - show inconclusive results'
-    print '  p - show imap results'
-    print '  o - show pop results'
-    print '  t - show smtp results'
-    print '  d - show dns results'
-    print '  r - show dnsrebind results'
-    print ''
-
-#
-# Displaying stats in a graphical setting (first check if we have wx)
-#
-
-nowx = False
-try:
-  import wx
-  from wx.lib.mixins.listctrl import ListCtrlAutoWidthMixin, ColumnSorterMixin
-except:
-  nowx = True
-
-if not nowx:
-
-  class ListMixin(wx.ListCtrl, ListCtrlAutoWidthMixin, ColumnSorterMixin):
-    def __init__(self, parent, map):
-      wx.ListCtrl.__init__(self, parent, -1, style=wx.LC_REPORT)
-      ListCtrlAutoWidthMixin.__init__(self)
-      ColumnSorterMixin.__init__(self, len(map))
-      self.itemDataMap = map
-
-    def GetListCtrl(self):
-      return self
-
-  # menu item ids
-  ID_EXIT = 1
-
-  ID_SHOW_GOOD = 11
-  ID_SHOW_BAD = 12
-  ID_SHOW_UNSURE = 13
-
-  ID_SHOW_SSL = 21
-  ID_SHOW_HTTP = 22
-  ID_SHOW_SSH = 23
-  ID_SHOW_SMTP = 24
-  ID_SHOW_IMAP = 25
-  ID_SHOW_POP = 26
-  ID_SHOW_DNS = 27
-  ID_SHOW_DNSREBIND = 28
-
-  ID_NODE = 31
-
-  class MainFrame(wx.Frame):
-    ''' the main application window for displaying statistics with a GUI'''
-    def __init__(self):
-      wx.Frame.__init__(self, None, title="Soat test results", size=(900,500))
-     
-      # get the data
-
-      self.dataHandler = DataHandler()
-      self.dataList = self.dataHandler.getAll()
-      self.filteredList = self.dataList
-
-      # display it
-    
-      self.CreateStatusBar()
-      self.initMenuBar()
-      self.initContent()
-
-      self.Center()
-      self.Show()
-  
-    def initMenuBar(self):
-      fileMenu = wx.Menu()
-      fileMenu.Append(ID_EXIT, "E&xit", "Exit the program")
-    
-      viewMenu = wx.Menu()
-      self.showGood = viewMenu.Append(ID_SHOW_GOOD, 'Show &Good', 'Show sucessful test results', kind=wx.ITEM_CHECK)
-      self.showBad = viewMenu.Append(ID_SHOW_BAD, 'Show &Bad', 'Show unsucessful test results', kind=wx.ITEM_CHECK)
-      self.showUnsure = viewMenu.Append(ID_SHOW_UNSURE, 'Show &Inconclusive', 'Show inconclusive test results', kind=wx.ITEM_CHECK)
-      viewMenu.AppendSeparator()
-      self.showSSL = viewMenu.Append(ID_SHOW_SSL, 'Show SS&L', 'Show SSL test results', kind=wx.ITEM_CHECK)
-      self.showHTTP = viewMenu.Append(ID_SHOW_HTTP, 'Show &HTTP', 'Show HTTP test results', kind=wx.ITEM_CHECK)
-      self.showSSH = viewMenu.Append(ID_SHOW_SSH, 'Show &SSH', 'Show SSH test results', kind=wx.ITEM_CHECK)
-      viewMenu.AppendSeparator()
-      self.showSMTP = viewMenu.Append(ID_SHOW_SMTP, 'Show SMTP', 'Show SMTP test results', kind=wx.ITEM_CHECK)
-      self.showIMAP = viewMenu.Append(ID_SHOW_IMAP, 'Show IMAP', 'Show IMAP test results', kind=wx.ITEM_CHECK)
-      self.showPOP = viewMenu.Append(ID_SHOW_POP, 'Show POP', 'Show POP test results', kind=wx.ITEM_CHECK)
-      viewMenu.AppendSeparator()
-      self.showDNS = viewMenu.Append(ID_SHOW_DNS, 'Show DNS', 'Show DNS test results', kind=wx.ITEM_CHECK)
-      self.showDNSRebind = viewMenu.Append(ID_SHOW_DNSREBIND, 'Show DNSRebind', 'Show DNS rebind test results', kind=wx.ITEM_CHECK)
-      viewMenu.AppendSeparator()
-      viewMenu.Append(ID_NODE, '&Find node...', 'View test results for a given node [NOT IMPLEMENTED]')
-  
-      menuBar = wx.MenuBar()
-      menuBar.Append(fileMenu,"&File")
-      menuBar.Append(viewMenu,"&View")
-
-      self.SetMenuBar(menuBar)
-
-      wx.EVT_MENU(self, ID_EXIT, self.OnExit)
-
-      wx.EVT_MENU(self, ID_SHOW_GOOD, self.GenerateFilteredList)
-      wx.EVT_MENU(self, ID_SHOW_BAD, self.GenerateFilteredList)
-      wx.EVT_MENU(self, ID_SHOW_UNSURE, self.GenerateFilteredList)
-      viewMenu.Check(ID_SHOW_GOOD, True)
-      viewMenu.Check(ID_SHOW_BAD, True)
-      viewMenu.Check(ID_SHOW_UNSURE, True)
-      
-      for i in range(ID_SHOW_SSL, ID_SHOW_DNSREBIND + 1):
-        viewMenu.Check(i, True)
-        wx.EVT_MENU(self, i, self.GenerateFilteredList)
-
-    def initContent(self): 
-      base = wx.Panel(self, -1)
-      sizer = wx.GridBagSizer(0,0)
-
-      box = wx.StaticBox(base, -1, 'Summary')
-      boxSizer = wx.StaticBoxSizer(box, wx.HORIZONTAL)
-
-      total = wx.StaticText(base, -1, 'Total tests: ' + `len(self.filteredList)`)
-      boxSizer.Add(total, 0, wx.LEFT | wx.TOP | wx.BOTTOM, 10)
-
-      nodes = wx.StaticText(base, -1, 'Nodes scanned: ' + `len(Set([x.exit_node for x in self.filteredList]))`)
-      boxSizer.Add(nodes, 0, wx.LEFT | wx.TOP | wx.BOTTOM , 10)
-
-      bad = wx.StaticText(base, -1, 'Failed tests: ' + `len([x for x in self.filteredList if x.status == 2])`)
-      boxSizer.Add(bad, 0, wx.LEFT | wx.TOP | wx.BOTTOM, 10)
-
-      suspicious = wx.StaticText(base, -1, 'Inconclusive tests: ' + `len([x for x in self.filteredList if x.status == 1])`)
-      boxSizer.Add(suspicious, 0, wx.ALL, 10)
-
-      sizer.Add(boxSizer, (0,0), (1, 5), wx.EXPAND | wx.ALL, 15)
-
-      dataMap = {}
-      self.fillDataMap(dataMap)
-    
-      self.listCtrl = ListMixin(base, dataMap)
-      self.listCtrl.InsertColumn(0, 'exit node', width=380)
-      self.listCtrl.InsertColumn(1, 'type', width=70)
-      self.listCtrl.InsertColumn(2, 'site', width=180)
-      self.listCtrl.InsertColumn(3, 'time', width=180)
-      self.listCtrl.InsertColumn(4, 'status', wx.LIST_FORMAT_CENTER, width=50)
-
-      self.fillListCtrl(dataMap)
-    
-      sizer.Add(self.listCtrl, (1,0), (1,5), wx.EXPAND | wx.LEFT | wx.BOTTOM | wx.RIGHT, border=15)
-
-      sizer.AddGrowableCol(3)
-      sizer.AddGrowableRow(1)
-
-      base.SetSizerAndFit(sizer)
-
-    # make a nasty dictionary from the current self.filteredList object so columns would be sortable
-    def fillDataMap(self, dataMap):
-      for i in range(len(self.filteredList)):
-        dataMap.update([(i,(self.filteredList[i].exit_node, 
-                self.filteredList[i].__class__.__name__[:-10],
-                self.filteredList[i].site, 
-                time.strftime("%a, %d %b %Y %H:%M:%S", time.localtime(self.filteredList[i].timestamp)), 
-                self.filteredList[i].status))])
-
-    # fill the result listing with data
-    def fillListCtrl(self, dataMap):
-      if self.listCtrl.GetItemCount() > 0:
-        self.listCtrl.DeleteAllItems()
-
-      for k, i in dataMap.items():
-        index = self.listCtrl.InsertStringItem(sys.maxint, `i[0]`)
-        self.listCtrl.SetStringItem(index, 1, i[1])
-        self.listCtrl.SetStringItem(index, 2, `i[2]`) 
-        self.listCtrl.SetStringItem(index, 3, i[3])
-        self.listCtrl.SetStringItem(index, 4, `i[4]`)
-        self.listCtrl.SetItemData(index,k)
-
-    def OnExit(self,e):
-      self.Close(True)
-
-    def GenerateFilteredList(self, e): 
-      protocols = []
-      if self.showSSH.IsChecked():
-        protocols.append("ssh") 
-      if self.showHTTP.IsChecked():
-        protocols.append("http")
-      if self.showSSL.IsChecked():
-        protocols.append("ssl")
-      if self.showIMAP.IsChecked():
-        protocols.append("imap")
-      if self.showPOP.IsChecked():
-        protocols.append("pop")
-      if self.showSMTP.IsChecked():
-        protocols.append("smtp")
-      if self.showDNS.IsChecked():
-        protocols.append("dns")
-      if self.showDNSRebind.IsChecked():
-        protocols.append("dnsrebind")
-
-      self.filteredList = list(self.dataHandler.filterResults(self.dataList, protocols, 
-        self.showGood.IsChecked(), self.showBad.IsChecked(), self.showUnsure.IsChecked()))
-
-      dataMap = {}
-      self.fillDataMap(dataMap)
-      self.fillListCtrl(dataMap)
-      self.listCtrl.RefreshItems(0, len(dataMap)) 
-
-if __name__ == "__main__":
-  if len(sys.argv) == 1:
-    console = StatsConsole()
-    console.Listen()
-  elif len(sys.argv) == 2 and sys.argv[1] == 'wx':
-    if nowx:
-      print 'wxpython doesn\'t seem to be installed on your system'
-      print 'you can use the console interface instead (see help)'
-    else:
-      app = wx.App(0)
-      MainFrame()
-      app.MainLoop()
-  else:
-    print ''
-    print 'This app displays results of tests carried out by soat.py (in a user-friendly way).'
-    print ''
-    print 'Usage:'
-    print 'python soatstats.py - app starts console-only'
-    print 'python soatstats.py wx - app starts with a wxpython gui'
-    print ''