commit 6f7eac497a9d4062ff18d6a0b12791099e977f2c Author: Damian Johnson atagar@torproject.org Date: Sat Aug 1 16:51:27 2015 -0700
Move connection type to be tracked by entity
Our get_type() method is a method on individual lines rather than the parent entity, which is weird. All lines always share the same type, and we almost always fetch it from entites via...
my_entity.get_lines()[0].get_type()
Moving it to be tracked on the entity instead, and lines just report whatever its parent has. --- nyx/connections/circ_entry.py | 16 +++----- nyx/connections/conn_entry.py | 81 ++--------------------------------------- nyx/connections/conn_panel.py | 10 ++--- nyx/connections/entries.py | 73 ++++++++++++++++++++++++++++++++----- 4 files changed, 77 insertions(+), 103 deletions(-)
diff --git a/nyx/connections/circ_entry.py b/nyx/connections/circ_entry.py index 94a0196..7d3fc8d 100644 --- a/nyx/connections/circ_entry.py +++ b/nyx/connections/circ_entry.py @@ -24,8 +24,8 @@ class CircHeaderLine(conn_entry.ConnectionLine): lines except that its etc field has circuit attributes. """
- def __init__(self, circ): - conn_entry.ConnectionLine.__init__(self, nyx.util.tracker.Connection(entries.to_unix_time(circ.created), False, '127.0.0.1', 0, '0.0.0.0', 0, 'tcp'), False, False) + def __init__(self, entry, circ): + conn_entry.ConnectionLine.__init__(self, entry, nyx.util.tracker.Connection(entries.to_unix_time(circ.created), False, '127.0.0.1', 0, '0.0.0.0', 0, 'tcp'), False, False) self.circuit_id = circ.id self.purpose = circ.purpose.capitalize() self.is_built = False @@ -33,7 +33,7 @@ class CircHeaderLine(conn_entry.ConnectionLine): self._remote_fingerprint = None
def set_exit(self, exit_address, exit_port, exit_fingerprint): - conn_entry.ConnectionLine.__init__(self, nyx.util.tracker.Connection(self._timestamp, False, '127.0.0.1', 0, exit_address, exit_port, 'tcp'), False, False) + conn_entry.ConnectionLine.__init__(self, self._entry, nyx.util.tracker.Connection(self._timestamp, False, '127.0.0.1', 0, exit_address, exit_port, 'tcp'), False, False) self.is_built = True self._remote_fingerprint = exit_fingerprint
@@ -43,9 +43,6 @@ class CircHeaderLine(conn_entry.ConnectionLine): else: return conn_entry.ConnectionLine.get_fingerprint(self, default)
- def get_type(self): - return conn_entry.Category.CIRCUIT - def get_destination_label(self, max_length, include_locale = False): if not self.is_built: return 'Building...' @@ -83,8 +80,8 @@ class CircLine(conn_entry.ConnectionLine): caching, etc). """
- def __init__(self, remote_address, remote_port, remote_fingerprint, placement_label, timestamp): - conn_entry.ConnectionLine.__init__(self, nyx.util.tracker.Connection(timestamp, False, '127.0.0.1', 0, remote_address, remote_port, 'tcp')) + def __init__(self, entry, remote_address, remote_port, remote_fingerprint, placement_label, timestamp): + conn_entry.ConnectionLine.__init__(self, entry, nyx.util.tracker.Connection(timestamp, False, '127.0.0.1', 0, remote_address, remote_port, 'tcp')) self._remote_fingerprint = remote_fingerprint self.placement_label = placement_label self.include_port = False @@ -96,9 +93,6 @@ class CircLine(conn_entry.ConnectionLine): def get_fingerprint(self, default = None): self._remote_fingerprint
- def get_type(self): - return conn_entry.Category.CIRCUIT - def get_listing_prefix(self): if self.is_last: return (ord(' '), curses.ACS_LLCORNER, curses.ACS_HLINE, ord(' ')) diff --git a/nyx/connections/conn_entry.py b/nyx/connections/conn_entry.py index f8593ff..381afec 100644 --- a/nyx/connections/conn_entry.py +++ b/nyx/connections/conn_entry.py @@ -11,7 +11,6 @@ import nyx.util.ui_tools from nyx.util import tor_controller from nyx.connections import entries
-from stem.control import Listener from stem.util import conf, connection, enum, str_tools
# Connection Categories: @@ -59,13 +58,12 @@ class ConnectionLine(entries.ConnectionPanelLine): Display component of the ConnectionEntry. """
- def __init__(self, conn, include_port=True, include_expanded_addresses=True): + def __init__(self, entry, conn, include_port=True, include_expanded_addresses=True): entries.ConnectionPanelLine.__init__(self)
+ self._entry = entry self.connection = conn
- self.cached_type = None - # includes the port or expanded ip address field when displaying listing # information if true
@@ -225,80 +223,7 @@ class ConnectionLine(entries.ConnectionPanelLine): return False
def get_type(self): - """ - Provides our best guess at the current type of the connection. This - depends on consensus results, our current client circuits, etc. Results - are cached until this entry's display is reset. - """ - - # caches both to simplify the calls and to keep the type consistent until - # we want to reflect changes - - if not self.cached_type: - controller = tor_controller() - - my_hidden_service_ports = [] # ports belonging to our hidden service configuation - - for hs_config in controller.get_hidden_service_conf({}).values(): - my_hidden_service_ports += [entry[2] for entry in hs_config['HiddenServicePort']] - - if self.connection.local_port in controller.get_ports(Listener.OR, []): - base_type = Category.INBOUND - elif self.connection.local_port in controller.get_ports(Listener.DIR, []): - base_type = Category.INBOUND - elif self.connection.local_port in controller.get_ports(Listener.SOCKS, []): - base_type = Category.SOCKS - elif self.connection.remote_port in my_hidden_service_ports: - base_type = Category.HIDDEN - elif self.connection.local_port in controller.get_ports(Listener.CONTROL, []): - base_type = Category.CONTROL - else: - # Currently the only non-static categories are OUTBOUND vs... - # - EXIT since this depends on the current consensus - # - CIRCUIT if this is likely to belong to our guard usage - # - DIRECTORY if this is a single-hop circuit (directory mirror?) - # - # The exitability, circuits, and fingerprints are all cached by the - # tor_tools util keeping this a quick lookup. - - base_type = Category.OUTBOUND - destination_fingerprint = nyx.util.tracker.get_consensus_tracker().get_relay_fingerprint(self.connection.remote_address, self.connection.remote_port) - - if not destination_fingerprint: - # Not a known relay. This might be an exit connection. - - exit_policy = controller.get_exit_policy(None) - port = self.connection.remote_port if self.connection.remote_port else None - - if exit_policy and exit_policy.can_exit_to(self.connection.remote_address, port): - self.cached_type = Category.EXIT - else: - # This belongs to a known relay. If we haven't eliminated ourselves as - # a possible client or directory connection then check if it still - # holds true. - - my_circuits = controller.get_circuits([]) - - # Checks that this belongs to the first hop in a circuit that's - # either unestablished or longer than a single hop (ie, anything but - # a built 1-hop connection since those are most likely a directory - # mirror). - - for circ in my_circuits: - if circ.path and circ.path[0][0] == destination_fingerprint and (circ.status != 'BUILT' or len(circ.path) > 1): - self.cached_type = Category.CIRCUIT # matched a probable guard connection - - if not self.cached_type: - # Checks if we match a built, single hop circuit. - - for circ in my_circuits: - if circ.path and circ.path[0][0] == destination_fingerprint and circ.status == 'BUILT' and len(circ.path) == 1: - self.cached_type = Category.DIRECTORY - - if not self.cached_type: - self.cached_type = base_type - - return self.cached_type + return self._entry.get_type()
def get_etc_content(self, width, listing_type): """ diff --git a/nyx/connections/conn_panel.py b/nyx/connections/conn_panel.py index c16f874..7d06abe 100644 --- a/nyx/connections/conn_panel.py +++ b/nyx/connections/conn_panel.py @@ -438,7 +438,7 @@ class ConnectionPanel(panel.Panel, threading.Thread): elif not entries: title = 'Connections:' else: - counts = collections.Counter([entry.get_lines()[0].get_type() for entry in entries]) + counts = collections.Counter([entry.get_type() for entry in entries]) count_labels = ['%i %s' % (counts[category], category.lower()) for category in conn_entry.Category if counts[category]] title = 'Connections (%s):' % ', '.join(count_labels)
@@ -481,12 +481,12 @@ class ConnectionPanel(panel.Panel, threading.Thread): for entry in new_entries: entry_line = entry.get_lines()[0]
- if entry_line.is_private() and entry_line.get_type() == conn_entry.Category.INBOUND: + if entry_line.is_private() and entry.get_type() == conn_entry.Category.INBOUND: client_locale = entry_line.get_locale(None)
if client_locale: self._client_locale_usage[client_locale] = self._client_locale_usage.get(client_locale, 0) + 1 - elif entry_line.get_type() == conn_entry.Category.EXIT: + elif entry.get_type() == conn_entry.Category.EXIT: exit_port = entry_line.connection.remote_port self._exit_port_usage[exit_port] = self._exit_port_usage.get(exit_port, 0) + 1
@@ -501,9 +501,9 @@ class ConnectionPanel(panel.Panel, threading.Thread): for entry in new_entries: line = entry.get_lines()[0]
- if line.get_type() in (conn_entry.Category.SOCKS, conn_entry.Category.CONTROL): + if entry.get_type() in (conn_entry.Category.SOCKS, conn_entry.Category.CONTROL): local_ports.append(line.connection.remote_port) - elif line.get_type() == conn_entry.Category.HIDDEN: + elif entry.get_type() == conn_entry.Category.HIDDEN: remote_ports.append(line.connection.local_port)
nyx.util.tracker.get_port_usage_tracker().query(local_ports, remote_ports) diff --git a/nyx/connections/entries.py b/nyx/connections/entries.py index fa4e71f..7ea3321 100644 --- a/nyx/connections/entries.py +++ b/nyx/connections/entries.py @@ -6,6 +6,9 @@ consists of in the listing.
import datetime
+from nyx.util import tor_controller + +from stem.control import Listener from stem.util import enum
# attributes we can list entries by @@ -39,25 +42,27 @@ def to_unix_time(dt):
class ConnectionPanelEntry: - def __init__(self, start_time): + def __init__(self, connection_type, start_time): self.lines = [] - self.start_time = start_time + self._connection_type = connection_type + self._start_time = start_time
@staticmethod def from_connection(conn): import nyx.connections.conn_entry
- entry = ConnectionPanelEntry(conn.start_time) - entry.lines = [nyx.connections.conn_entry.ConnectionLine(conn)] + entry = ConnectionPanelEntry(get_type(conn), conn.start_time) + entry.lines = [nyx.connections.conn_entry.ConnectionLine(entry, conn)] return entry
@staticmethod def from_circuit(circ): import nyx.connections.circ_entry + import nyx.connections.conn_entry import nyx.util.tracker
- entry = ConnectionPanelEntry(to_unix_time(circ.created)) - entry.lines = [nyx.connections.circ_entry.CircHeaderLine(circ)] + entry = ConnectionPanelEntry(nyx.connections.conn_entry.Category.CIRCUIT, to_unix_time(circ.created)) + entry.lines = [nyx.connections.circ_entry.CircHeaderLine(entry, circ)]
path = [path_entry[0] for path_entry in circ.path]
@@ -77,12 +82,20 @@ class ConnectionPanelEntry:
placement_label = '%i / %s' % (i + 1, placement_type)
- entry.lines.append(nyx.connections.circ_entry.CircLine(relay_ip, relay_port, relay_fingerprint, placement_label, to_unix_time(circ.created))) + entry.lines.append(nyx.connections.circ_entry.CircLine(entry, relay_ip, relay_port, relay_fingerprint, placement_label, to_unix_time(circ.created)))
entry.lines[-1].is_last = True
return entry
+ def get_type(self): + """ + Provides our best guess at the current type of the connection. This + depends on consensus results, our current client circuits, etc. + """ + + return self._connection_type + def get_lines(self): """ Provides the individual lines in the connection listing. @@ -128,9 +141,9 @@ class ConnectionPanelEntry: return 'z' * 20 # orders at the end elif attr == SortAttr.CATEGORY: import nyx.connections.conn_entry - return nyx.connections.conn_entry.Category.index_of(connection_line.get_type()) + return nyx.connections.conn_entry.Category.index_of(self.get_type()) elif attr == SortAttr.UPTIME: - return self.start_time + return self._start_time elif attr == SortAttr.COUNTRY: if connection_line.connection.is_private_address(self.lines[0].connection.remote_address): return '' @@ -214,3 +227,45 @@ class ConnectionPanelLine: def _get_details(self, width): # implementation of get_details return [] + + +def get_type(connection): + import nyx.connections.conn_entry + import nyx.util.tracker + + controller = tor_controller() + + my_hidden_service_ports = [] # ports belonging to our hidden service configuation + + for hs_config in controller.get_hidden_service_conf({}).values(): + my_hidden_service_ports += [entry[2] for entry in hs_config['HiddenServicePort']] + + if connection.local_port in controller.get_ports(Listener.OR, []): + return nyx.connections.conn_entry.Category.INBOUND + elif connection.local_port in controller.get_ports(Listener.DIR, []): + return nyx.connections.conn_entry.Category.INBOUND + elif connection.local_port in controller.get_ports(Listener.SOCKS, []): + return nyx.connections.conn_entry.Category.SOCKS + elif connection.remote_port in my_hidden_service_ports: + return nyx.connections.conn_entry.Category.HIDDEN + elif connection.local_port in controller.get_ports(Listener.CONTROL, []): + return nyx.connections.conn_entry.Category.CONTROL + else: + destination_fingerprint = nyx.util.tracker.get_consensus_tracker().get_relay_fingerprint(connection.remote_address, connection.remote_port) + + if not destination_fingerprint: + # Not a known relay. This might be an exit connection. + + exit_policy = controller.get_exit_policy(None) + + if exit_policy and exit_policy.can_exit_to(connection.remote_address, connection.remote_port): + return nyx.connections.conn_entry.Category.EXIT + else: + for circ in controller.get_circuits([]): + if circ.path[0][0] == destination_fingerprint and circ.status == 'BUILT': + # Tor builds one-hop circuits to retrieve directory information. + # If longer this is likely a connection to a guard. + + return nyx.connections.conn_entry.Category.DIRECTORY if len(circ.path) == 1 else nyx.connections.conn_entry.Category.CIRCUIT + + return nyx.connections.conn_entry.Category.OUTBOUND