[tor-commits] [chutney/master] Add 'verify' command.
nickm at torproject.org
nickm at torproject.org
Mon Jun 24 17:08:21 UTC 2013
commit 8d75181d9c2a45b555f27e5126173026987ddf0d
Author: Linus Nordberg <linus at torproject.org>
Date: Wed Jun 5 15:48:57 2013 +0200
Add 'verify' command.
Addresses #8531.
---
lib/chutney/TorNet.py | 27 +++++
lib/chutney/Traffic.py | 279 ++++++++++++++++++++++++++++++++++++++++++++++++
2 files changed, 306 insertions(+)
diff --git a/lib/chutney/TorNet.py b/lib/chutney/TorNet.py
index dcaf7d8..3ec33b6 100644
--- a/lib/chutney/TorNet.py
+++ b/lib/chutney/TorNet.py
@@ -1,6 +1,7 @@
#!/usr/bin/python
#
# Copyright 2011 Nick Mathewson, Michael Stone
+# Copyright 2013 The Tor Project
#
# You may do anything with this work that copyright law would normally
# restrict, so long as you retain the above notice(s) and this license
@@ -21,6 +22,7 @@ import errno
import time
import chutney.Templating
+import chutney.Traffic
def mkdir_p(d, mode=0777):
"""Create directory 'd' and all of its parents as needed. Unlike
@@ -621,6 +623,31 @@ class Network(object):
for c in controllers:
n.check(listNonRunning=False)
+ def verify(self):
+ sys.stdout.write("Verifying data transmission: ")
+ sys.stdout.flush()
+ status = self._verify_traffic()
+ if status:
+ print("Success")
+ else:
+ print("Failure")
+ return status
+
+ def _verify_traffic(self):
+ """Verify (parts of) the network by sending traffic through it
+ and verify what is received."""
+ LISTEN_PORT = 4747 # FIXME: Do better! Note the default exit policy.
+ DATALEN = 10*1024 # Octets.
+ TIMEOUT = 3 # Seconds.
+ with open('/dev/urandom', 'r') as randfp:
+ tmpdata = randfp.read(DATALEN)
+ bind_to = ('localhost', LISTEN_PORT)
+ tt = chutney.Traffic.TrafficTester(bind_to, tmpdata, TIMEOUT)
+ for op in filter(lambda n: n._env['tag'] == 'c', self._nodes):
+ tt.add(chutney.Traffic.Source(tt, bind_to, tmpdata,
+ ('localhost', int(op._env['socksport']))))
+ return tt.run()
+
def ConfigureNodes(nodelist):
network = _THE_NETWORK
diff --git a/lib/chutney/Traffic.py b/lib/chutney/Traffic.py
new file mode 100755
index 0000000..4ab4fff
--- /dev/null
+++ b/lib/chutney/Traffic.py
@@ -0,0 +1,279 @@
+#! /usr/bin/env python
+#
+# Copyright 2013 The Tor Project
+#
+# You may do anything with this work that copyright law would normally
+# restrict, so long as you retain the above notice(s) and this license
+# in all redistributed copies and derived works. There is no warranty.
+
+import socket
+import select
+import struct
+
+debug_flag = False
+
+def debug(s):
+ if debug_flag:
+ print("DEBUG: %s" % s)
+
+def socks_cmd(addr_port):
+ """
+ SOCKSv4: https://en.wikipedia.org/wiki/SOCKS#Protocol
+ SOCKSv5: RFC1928, RFC1929
+ """
+ ver = 4 # Only SOCKSv4 for now.
+ cmd = 1 # Stream connection.
+ user = '\x00'
+ dnsname = ''
+ host, port = addr_port
+ try:
+ addr = socket.inet_aton(host)
+ except socket.error:
+ addr = '\x00\x00\x00\x01'
+ dnsname = '%s\x00' % host
+ return struct.pack('!BBH', ver, cmd, port) + addr + user + dnsname
+
+class TestSuite(object):
+ """Keep a tab on how many tests are pending, how many have failed
+ and how many have succeeded."""
+ def __init__(self):
+ self.not_done = 0
+ self.successes = 0
+ self.failures = 0
+
+ def add(self):
+ self.not_done += 1
+
+ def success(self):
+ self.not_done -= 1
+ self.successes += 1
+
+ def failure(self):
+ self.not_done -= 1
+ self.failures += 1
+
+ def failure_count(self):
+ return self.failures
+
+ def all_done(self):
+ return self.not_done == 0
+
+ def status(self):
+ return('%d/%d/%d' % (self.not_done, self.successes, self.failures))
+
+class Peer(object):
+ LISTENER = 1
+ SOURCE = 2
+ SINK = 3
+
+ def __init__(self, ptype, tt, s=None):
+ self.type = ptype
+ self.tt = tt # TrafficTester
+ if s is not None:
+ self.s = s
+ else:
+ self.s = socket.socket()
+ self.s.setblocking(False)
+
+ def fd(self):
+ return self.s.fileno()
+ def is_source(self):
+ return self.type == self.SOURCE
+ def is_sink(self):
+ return self.type == self.SINK
+
+class Listener(Peer):
+ def __init__(self, tt, endpoint):
+ super(Listener, self).__init__(Peer.LISTENER, tt)
+ self.s.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
+ self.s.bind(endpoint)
+ self.s.listen(0)
+
+ def accept(self):
+ newsock, endpoint = self.s.accept()
+ debug("new client from %s:%s (fd=%d)" %
+ (endpoint[0], endpoint[1], newsock.fileno()))
+ self.tt.add(Sink(self.tt, newsock))
+
+class Sink(Peer):
+ def __init__(self, tt, s):
+ super(Sink, self).__init__(Peer.SINK, tt, s)
+ self.inbuf = ''
+
+ def on_readable(self):
+ return self.verify(self.tt.data)
+
+ def verify(self, data):
+ self.inbuf += self.s.recv(len(data) - len(self.inbuf))
+ assert(len(self.inbuf) <= len(data))
+ if len(self.inbuf) == len(data):
+ if self.inbuf != data:
+ return -1 # Failed verification.
+ else:
+ debug("successful verification")
+ return len(data) - len(self.inbuf)
+
+class Source(Peer):
+ NOT_CONNECTED = 0
+ CONNECTING = 1
+ CONNECTING_THROUGH_PROXY = 2
+ CONNECTED = 5
+
+ def __init__(self, tt, server, buf, proxy=None):
+ super(Source, self).__init__(Peer.SOURCE, tt)
+ self.state = self.NOT_CONNECTED
+ self.data = buf
+ self.outbuf = ''
+ self.inbuf = ''
+ self.proxy = proxy
+ self.connect(server)
+
+ def connect(self, endpoint):
+ self.dest = endpoint
+ self.state = self.CONNECTING
+ if self.proxy is None:
+ dest = self.dest
+ else:
+ dest = self.proxy
+ try:
+ self.s.connect(dest)
+ except socket.error, e:
+ if e[0] != 115: # EINPROGRESS
+ raise
+
+ def on_readable(self):
+ if self.state == self.CONNECTING_THROUGH_PROXY:
+ self.inbuf += self.s.recv(8 - len(self.inbuf))
+ if len(self.inbuf) == 8:
+ if ord(self.inbuf[0]) == 0 and ord(self.inbuf[1]) == 0x5a:
+ debug("proxy handshake successful (fd=%d)" % self.fd())
+ self.state = self.CONNECTED
+ self.inbuf = ''
+ self.outbuf = self.data
+ else:
+ debug("proxy handshake failed (0x%x)! (fd=%d)" %
+ (ord(self.inbuf[1]), self.fd()))
+ self.state = self.NOT_CONNECTED
+ return -1
+ return 8 - len(self.inbuf)
+ return 1 # Keep us around for writing.
+
+ def want_to_write(self):
+ if self.state == self.CONNECTING:
+ return True
+ if len(self.outbuf) > 0:
+ return True
+ return False
+
+ def on_writable(self):
+ if self.state == self.CONNECTING:
+ if self.proxy is None:
+ self.state = self.CONNECTED
+ self.outbuf = self.data
+ else:
+ self.state = self.CONNECTING_THROUGH_PROXY
+ self.outbuf = socks_cmd(self.dest)
+ try:
+ n = self.s.send(self.outbuf)
+ except socket.error, e:
+ if e[0] == 111: # ECONNREFUSED
+ debug("connection refused (fd=%d)" % self.fd())
+ return -1
+ raise
+ self.outbuf = self.outbuf[n:]
+ debug("successfully connected (fd=%d)" % self.fd())
+ if self.state == self.CONNECTING_THROUGH_PROXY:
+ return 1 # Keep us around.
+ return len(self.outbuf) # When 0, we're being removed.
+
+class TrafficTester():
+ def __init__(self, endpoint, data={}, timeout=3):
+ self.listener = Listener(self, endpoint)
+ self.pending_close = []
+ self.timeout = timeout
+ self.tests = TestSuite()
+ self.data = data
+ debug("listener fd=%d" % self.listener.fd())
+ self.peers = {} # fd:Peer
+
+ def sinks(self):
+ return self.get_by_ptype(Peer.SINK)
+ def sources(self):
+ return self.get_by_ptype(Peer.SOURCE)
+ def get_by_ptype(self, ptype):
+ return filter(lambda p: p.type == ptype, self.peers.itervalues())
+
+ def add(self, peer):
+ self.peers[peer.fd()] = peer
+ if peer.is_source():
+ self.tests.add()
+
+ def remove(self, peer):
+ self.peers.pop(peer.fd())
+ self.pending_close.append(peer.s)
+
+ def run(self):
+ while True:
+ if self.tests.all_done() or self.timeout == 0:
+ break
+ rset = [self.listener.fd()] + list(self.peers)
+ wset = [p.fd() for p in
+ filter(lambda x: x.want_to_write(), self.sources())]
+ #debug("rset %s wset %s" % (rset, wset))
+ sets = select.select(rset, wset, [], 1)
+ if all(len(s)==0 for s in sets):
+ self.timeout -= 1
+ continue
+
+ for fd in sets[0]: # readable fd's
+ if fd == self.listener.fd():
+ self.listener.accept()
+ continue
+ p = self.peers[fd]
+ n = p.on_readable()
+ if n > 0:
+ #debug("need %d more octets from fd %d" % (n, fd))
+ pass
+ elif n == 0: # Success.
+ self.tests.success()
+ self.remove(p)
+ else: # Failure.
+ self.tests.failure()
+ if p.is_sink():
+ print("verification failed!")
+ self.remove(p)
+
+ for fd in sets[1]: # writable fd's
+ p = self.peers.get(fd) # Might have been removed above.
+ if p is not None:
+ n = p.on_writable()
+ if n == 0:
+ self.remove(p)
+ elif n < 0:
+ self.tests.failure()
+ self.remove(p)
+
+ self.listener.s.close()
+ for s in self.pending_close:
+ s.close()
+ return self.tests.all_done() and self.tests.failure_count() == 0
+
+import sys
+
+def main():
+ """Test the TrafficTester by sending and receiving some data."""
+ DATA = "a foo is a bar" * 1000
+ DATA_CORRUPT = "a foo is a baz" * 1000
+ proxy = ('localhost', 9008)
+ bind_to = ('localhost', int(sys.argv[1]))
+
+ tt = TrafficTester(bind_to, DATA)
+ tt.add(Source(tt, bind_to, DATA, proxy))
+ success = tt.run()
+
+ if success:
+ return 0
+ return -1
+
+if __name__ == '__main__':
+ sys.exit(main())
More information about the tor-commits
mailing list