[tor-commits] [nyx/master] Test torrc panel

atagar at torproject.org atagar at torproject.org
Mon Jul 11 16:58:33 UTC 2016


commit f388d3b984001207bca9c423d5340a3e44ea857c
Author: Damian Johnson <atagar at torproject.org>
Date:   Mon Jul 11 09:39:54 2016 -0700

    Test torrc panel
    
    This panel is tiny enough that I'm not gonna break its draw function up for
    now. Just testing the whole panel instead.
---
 nyx/panel/torrc.py       | 29 ++++++++++-------
 test/panel/__init__.py   |  1 +
 test/panel/connection.py | 50 +++++++++++++++++++---------
 test/panel/torrc.py      | 84 ++++++++++++++++++++++++++++++++++++++++++++++++
 4 files changed, 137 insertions(+), 27 deletions(-)

diff --git a/nyx/panel/torrc.py b/nyx/panel/torrc.py
index 278fbeb..868e546 100644
--- a/nyx/panel/torrc.py
+++ b/nyx/panel/torrc.py
@@ -17,6 +17,17 @@ from stem import ControllerError
 from stem.control import State
 
 
+def _read_torrc(path):
+  contents = []
+
+  with open(path) as torrc_file:
+    for line in torrc_file.readlines():
+      line = line.replace('\t', '   ').replace('\xc2', "'").rstrip()
+      contents.append(filter(lambda char: char in string.printable, line))
+
+  return contents
+
+
 class TorrcPanel(panel.Panel):
   """
   Renders the current torrc or nyxrc with syntax highlighting in a scrollable
@@ -47,20 +58,14 @@ class TorrcPanel(panel.Panel):
     if event_type == State.RESET:
       try:
         self._torrc_location = expand_path(controller.get_info('config-file'))
-        contents = []
-
-        with open(self._torrc_location) as torrc_file:
-          for line in torrc_file.readlines():
-            line = line.replace('\t', '   ').replace('\xc2', "'").rstrip()
-            contents.append(filter(lambda char: char in string.printable, line))
-
-        self._torrc_content = contents
+        self._torrc_content = _read_torrc(self._torrc_location)
       except ControllerError as exc:
         self._torrc_load_error = msg('panel.torrc.unable_to_find_torrc', error = exc)
         self._torrc_location = None
         self._torrc_content = None
       except Exception as exc:
-        self._torrc_load_error = msg('panel.torrc.unable_to_load_torrc', error = exc.strerror)
+        exc_msg = exc.strerror if (hasattr(exc, 'strerror') and exc.strerror) else str(exc)
+        self._torrc_load_error = msg('panel.torrc.unable_to_load_torrc', error = exc_msg)
         self._torrc_content = None
 
   def set_comments_visible(self, is_visible):
@@ -121,7 +126,7 @@ class TorrcPanel(panel.Panel):
 
       if self._last_content_height > subwindow.height - 1:
         scroll_offset = 3
-        subwindow.scrollbar(1, scroll, height - 1)
+        subwindow.scrollbar(1, scroll, subwindow.height - 1)
 
       y = 1 - scroll
       is_multiline = False  # true if we're in the middle of a multiline torrc entry
@@ -144,8 +149,8 @@ class TorrcPanel(panel.Panel):
         elif ' ' not in line.strip():
           option, argument = line, ''  # no argument
         else:
-          whitespace = ' ' * (len(line) - len(line.strip()))
-          option, argument = line.strip().split(' ', 1)
+          whitespace = ' ' * (len(line) - len(line.lstrip()))
+          option, argument = line.lstrip().split(' ', 1)
           option = whitespace + option + ' '
 
         is_multiline = line.endswith('\\')  # next line's part of a multi-line entry
diff --git a/test/panel/__init__.py b/test/panel/__init__.py
index f1a9d7d..9142e64 100644
--- a/test/panel/__init__.py
+++ b/test/panel/__init__.py
@@ -7,4 +7,5 @@ __all__ = [
   'graph',
   'log',
   'connection',
+  'torrc',
 ]
diff --git a/test/panel/connection.py b/test/panel/connection.py
index 632ced0..30db919 100644
--- a/test/panel/connection.py
+++ b/test/panel/connection.py
@@ -118,30 +118,40 @@ def line(entry = MockEntry(), line_type = LineType.CONNECTION, connection = CONN
 class TestConnectionPanel(unittest.TestCase):
   @require_curses
   def test_draw_title(self):
-    self.assertEqual('Connection Details:', test.render(nyx.panel.connection._draw_title, [], True).content)
-    self.assertEqual('Connections:', test.render(nyx.panel.connection._draw_title, [], False).content)
+    rendered = test.render(nyx.panel.connection._draw_title, [], True)
+    self.assertEqual('Connection Details:', rendered.content)
+
+    rendered = test.render(nyx.panel.connection._draw_title, [], False)
+    self.assertEqual('Connections:', rendered.content)
 
     entries = [MockEntry(entry_type = category) for category in (Category.INBOUND, Category.INBOUND, Category.OUTBOUND, Category.INBOUND, Category.CONTROL)]
-    self.assertEqual('Connections (3 inbound, 1 outbound, 1 control):', test.render(nyx.panel.connection._draw_title, entries, False).content)
+
+    rendered = test.render(nyx.panel.connection._draw_title, entries, False)
+    self.assertEqual('Connections (3 inbound, 1 outbound, 1 control):', rendered.content)
 
   @require_curses
   def test_draw_details_incomplete_circuit(self):
     selected = line(line_type = LineType.CIRCUIT_HEADER, circ = MockCircuit(status = 'EXTENDING'))
-    self.assertEqual(DETAILS_BUILDING_CIRCUIT, test.render(nyx.panel.connection._draw_details, selected).content)
+
+    rendered = test.render(nyx.panel.connection._draw_details, selected)
+    self.assertEqual(DETAILS_BUILDING_CIRCUIT, rendered.content)
 
   @require_curses
   @patch('nyx.tracker.get_consensus_tracker')
   def test_draw_details_no_consensus_data(self, consensus_tracker_mock):
     consensus_tracker_mock().get_relay_fingerprints.return_value = None
-    self.assertEqual(DETAILS_NO_CONSENSUS_DATA, test.render(nyx.panel.connection._draw_details, line()).content)
+
+    rendered = test.render(nyx.panel.connection._draw_details, line())
+    self.assertEqual(DETAILS_NO_CONSENSUS_DATA, rendered.content)
 
   @require_curses
   @patch('nyx.tracker.get_consensus_tracker')
   def test_draw_details_when_private(self, consensus_tracker_mock):
     consensus_tracker_mock().get_relay_fingerprints.return_value = None
-
     selected = line(entry = MockEntry(is_private = True))
-    self.assertEqual(DETAILS_WHEN_PRIVATE, test.render(nyx.panel.connection._draw_details, selected).content)
+
+    rendered = test.render(nyx.panel.connection._draw_details, selected)
+    self.assertEqual(DETAILS_WHEN_PRIVATE, rendered.content)
 
   @require_curses
   @patch('nyx.panel.connection.tor_controller')
@@ -168,7 +178,8 @@ class TestConnectionPanel(unittest.TestCase):
       22: 'B6D83EC2D9E18B0A7A33428F8CFA9C536769E209'
     }
 
-    self.assertEqual(DETAILS_FOR_RELAY, test.render(nyx.panel.connection._draw_details, line()).content)
+    rendered = test.render(nyx.panel.connection._draw_details, line())
+    self.assertEqual(DETAILS_FOR_RELAY, rendered.content)
 
   @require_curses
   @patch('nyx.tracker.get_consensus_tracker')
@@ -179,7 +190,8 @@ class TestConnectionPanel(unittest.TestCase):
       443: 'E0BD57A11F00041A9789577C53A1B784473669E4',
     }
 
-    self.assertEqual(DETAILS_FOR_MULTIPLE_MATCHES, test.render(nyx.panel.connection._draw_details, line()).content)
+    rendered = test.render(nyx.panel.connection._draw_details, line())
+    self.assertEqual(DETAILS_FOR_MULTIPLE_MATCHES, rendered.content)
 
   @require_curses
   @patch('nyx.panel.connection.tor_controller')
@@ -201,7 +213,8 @@ class TestConnectionPanel(unittest.TestCase):
     }
 
     for test_line, expected in test_data.items():
-      self.assertEqual(expected, test.render(nyx.panel.connection._draw_line, 0, 0, test_line, False, 80, TIMESTAMP + 15.4).content)
+      rendered = test.render(nyx.panel.connection._draw_line, 0, 0, test_line, False, 80, TIMESTAMP + 15.4)
+      self.assertEqual(expected, rendered.content)
 
   @require_curses
   @patch('nyx.panel.connection.tor_controller')
@@ -221,7 +234,8 @@ class TestConnectionPanel(unittest.TestCase):
     }
 
     for test_line, expected in test_data.items():
-      self.assertEqual(expected, test.render(nyx.panel.connection._draw_address_column, 0, 0, test_line, ()).content)
+      rendered = test.render(nyx.panel.connection._draw_address_column, 0, 0, test_line, ())
+      self.assertEqual(expected, rendered.content)
 
   @require_curses
   @patch('nyx.tracker.get_port_usage_tracker')
@@ -242,15 +256,19 @@ class TestConnectionPanel(unittest.TestCase):
     }
 
     for test_line, expected in test_data.items():
-      self.assertEqual(expected, test.render(nyx.panel.connection._draw_line_details, 0, 0, test_line, 80, ()).content)
+      rendered = test.render(nyx.panel.connection._draw_line_details, 0, 0, test_line, 80, ())
+      self.assertEqual(expected, rendered.content)
 
   @require_curses
   def test_draw_right_column(self):
-    self.assertEqual('  1.0m (INBOUND)', test.render(nyx.panel.connection._draw_right_column, 0, 0, line(), TIMESTAMP + 62, ()).content)
+    rendered = test.render(nyx.panel.connection._draw_right_column, 0, 0, line(), TIMESTAMP + 62, ())
+    self.assertEqual('  1.0m (INBOUND)', rendered.content)
 
     legacy_connection = Connection(TIMESTAMP, True, '127.0.0.1', 3531, '75.119.206.243', 22, 'tcp', False)
     test_line = line(entry = MockEntry(entry_type = Category.CONTROL), connection = legacy_connection)
-    self.assertEqual('+ 1.1m (CONTROL)', test.render(nyx.panel.connection._draw_right_column, 0, 0, test_line, TIMESTAMP + 68, ()).content)
+
+    rendered = test.render(nyx.panel.connection._draw_right_column, 0, 0, test_line, TIMESTAMP + 68, ())
+    self.assertEqual('+ 1.1m (CONTROL)', rendered.content)
 
     test_data = {
       '1F43EE37A0670301AD9CB555D94AFEC2C89FDE86': '    1 / Guard',
@@ -260,4 +278,6 @@ class TestConnectionPanel(unittest.TestCase):
 
     for fp, expected in test_data.items():
       test_line = line(line_type = LineType.CIRCUIT, fingerprint = fp)
-      self.assertEqual(expected, test.render(nyx.panel.connection._draw_right_column, 0, 0, test_line, TIMESTAMP + 62, ()).content)
+
+      rendered = test.render(nyx.panel.connection._draw_right_column, 0, 0, test_line, TIMESTAMP + 62, ())
+      self.assertEqual(expected, rendered.content)
diff --git a/test/panel/torrc.py b/test/panel/torrc.py
new file mode 100644
index 0000000..96d222c
--- /dev/null
+++ b/test/panel/torrc.py
@@ -0,0 +1,84 @@
+"""
+Unit tests for nyx.panel.torrc.
+"""
+
+import unittest
+
+import nyx.panel.torrc
+import test
+
+from test import require_curses
+from mock import patch, Mock
+
+TORRC = """
+ORPort 9050
+ControlPort 9051
+Exitpolicy reject *:*  # non-exit relay
+CookieAuthentication 1
+""".strip()
+
+RENDERED_DEFAULT = """
+Tor Configuration File (/path/to/torrc):
+1 ORPort 9050
+2 ControlPort 9051
+3 Exitpolicy reject *:*  # non-exit relay
+4 CookieAuthentication 1
+""".strip()
+
+RENDERED_WITHOUT_COMMENTS = """
+Tor Configuration File (/path/to/torrc):
+1 ORPort 9050
+2 ControlPort 9051
+3 Exitpolicy reject *:*
+4 CookieAuthentication 1
+""".strip()
+
+RENDERED_WITHOUT_LINE_NUMBERS = """
+Tor Configuration File (/path/to/torrc):
+ORPort 9050
+ControlPort 9051
+Exitpolicy reject *:*  # non-exit relay
+CookieAuthentication 1
+""".strip()
+
+RENDERED_WITH_ERROR = """
+Tor Configuration File (/path/to/torrc):
+Unable to read our torrc: [Errno 2] No such file or directory: '/path/to/torrc'
+""".strip()
+
+
+class TestGraphPanel(unittest.TestCase):
+  @require_curses
+  @patch('nyx.panel.torrc._read_torrc', Mock(return_value = TORRC.splitlines()))
+  @patch('nyx.panel.torrc.expand_path', Mock(return_value = '/path/to/torrc'))
+  @patch('nyx.panel.torrc.tor_controller', Mock())
+  def test_draw_with_content(self):
+    panel = nyx.panel.torrc.TorrcPanel()
+    self.assertEqual(RENDERED_DEFAULT, test.render(panel.draw).content)
+
+  @require_curses
+  @patch('nyx.panel.torrc._read_torrc', Mock(return_value = TORRC.splitlines()))
+  @patch('nyx.panel.torrc.expand_path', Mock(return_value = '/path/to/torrc'))
+  @patch('nyx.panel.torrc.tor_controller', Mock())
+  def test_draw_without_comments(self):
+    panel = nyx.panel.torrc.TorrcPanel()
+    panel._show_comments = False
+    self.assertEqual(RENDERED_WITHOUT_COMMENTS, test.render(panel.draw).content)
+
+  @require_curses
+  @patch('nyx.panel.torrc._read_torrc', Mock(return_value = TORRC.splitlines()))
+  @patch('nyx.panel.torrc.expand_path', Mock(return_value = '/path/to/torrc'))
+  @patch('nyx.panel.torrc.tor_controller', Mock())
+  def test_draw_without_line_numbers(self):
+    panel = nyx.panel.torrc.TorrcPanel()
+    panel._show_line_numbers = False
+    self.assertEqual(RENDERED_WITHOUT_LINE_NUMBERS, test.render(panel.draw).content)
+
+  @require_curses
+  @patch('nyx.panel.torrc._read_torrc', Mock(side_effect = IOError("[Errno 2] No such file or directory: '/path/to/torrc'")))
+  @patch('nyx.panel.torrc.expand_path', Mock(return_value = '/path/to/torrc'))
+  @patch('nyx.panel.torrc.tor_controller', Mock())
+  def test_draw_with_error(self):
+    panel = nyx.panel.torrc.TorrcPanel()
+    panel._show_line_numbers = False
+    self.assertEqual(RENDERED_WITH_ERROR, test.render(panel.draw).content)





More information about the tor-commits mailing list