commit 3dcc573e7aca39f6b12d74885d51b1d01b483f90
Author: Damian Johnson <atagar(a)torproject.org>
Date: Sun Jan 20 15:41:42 2019 -0800
BandwidthFile creation
Nothing fancy, just implementing Descriptor's create() and content() method.
---
stem/descriptor/bandwidth_file.py | 60 ++++++++++++++++++++-
test/unit/descriptor/bandwidth_file.py | 98 ++++++++++++++++++++++++++++++++--
2 files changed, 152 insertions(+), 6 deletions(-)
diff --git a/stem/descriptor/bandwidth_file.py b/stem/descriptor/bandwidth_file.py
index 97210d7b..0b2b211a 100644
--- a/stem/descriptor/bandwidth_file.py
+++ b/stem/descriptor/bandwidth_file.py
@@ -15,11 +15,14 @@ Parsing for Bandwidth Authority metrics as described in Tor's
"""
import datetime
+import time
import stem.util.str_tools
from stem.descriptor import Descriptor
+HEADER_DIV = '====='
+
# Converters header attributes to a given type. Malformed fields should be
# ignored according to the spec.
@@ -93,7 +96,7 @@ def _parse_header(descriptor, entries):
lines = lines[1:]
for line in lines:
- if line == '=====':
+ if line == HEADER_DIV:
break
elif line.startswith('node_id='):
break # version 1.0 measurement
@@ -155,8 +158,61 @@ class BandwidthFile(Descriptor):
ATTRIBUTES.update(dict([(k, (None, _parse_header)) for k in HEADER_ATTR.keys()]))
+ @classmethod
+ def content(cls, attr = None, exclude = (), sign = False):
+ """
+ Creates descriptor content with the given attributes. This descriptor type
+ differs somewhat from others and treats our attr/exclude attributes as
+ follows...
+
+ * 'timestamp' is a reserved key for our mandatory header unix timestamp.
+
+ * 'content' is a reserved key for a list of our bandwidth measurements.
+
+ * All other keys are treated as header fields.
+
+ For example...
+
+ ::
+
+ BandwidthFile.content({
+ 'timestamp': '12345',
+ 'version': '1.2.0',
+ 'content': [],
+ })
+ """
+
+ if sign:
+ raise NotImplementedError('Signing of %s not implemented' % cls.__name__)
+
+ header = dict(attr) if attr is not None else {}
+ timestamp = header.pop('timestamp', str(int(time.time())))
+ content = header.pop('content', [])
+ version = header.get('version', HEADER_DEFAULT.get('version'))
+
+ lines = []
+
+ if 'timestamp' not in exclude:
+ lines.append(timestamp)
+
+ if version == '1.0.0' and header:
+ raise ValueError('Headers require BandwidthFile version 1.1 or later')
+ elif version != '1.0.0':
+ for k, v in header.items():
+ lines.append('%s=%s' % (k, v))
+
+ lines.append(HEADER_DIV)
+
+ for measurement in content:
+ lines.append(measurement) # TODO: replace when we have a measurement struct
+
+ return '\n'.join(lines)
+
def __init__(self, raw_content, validate = False):
super(BandwidthFile, self).__init__(raw_content, lazy_load = not validate)
+ self.content = [] # TODO: implement
+
if validate:
- pass # TODO: implement eager load
+ _parse_timestamp(self, None)
+ _parse_header(self, None)
diff --git a/test/unit/descriptor/bandwidth_file.py b/test/unit/descriptor/bandwidth_file.py
index b8fdf01b..e1f8ffb4 100644
--- a/test/unit/descriptor/bandwidth_file.py
+++ b/test/unit/descriptor/bandwidth_file.py
@@ -6,9 +6,22 @@ import datetime
import unittest
import stem.descriptor
-import stem.descriptor.bandwidth_file
-import test.unit.descriptor
+from stem.descriptor.bandwidth_file import BandwidthFile
+from test.unit.descriptor import get_resource
+
+try:
+ # added in python 3.3
+ from unittest.mock import Mock, patch
+except ImportError:
+ from mock import Mock, patch
+
+EXPECTED_NEW_HEADER_CONTENT = """
+1410723598
+version=1.1.0
+new_header=neat stuff
+=====
+""".strip()
class TestBandwidthFile(unittest.TestCase):
@@ -17,7 +30,7 @@ class TestBandwidthFile(unittest.TestCase):
Parse version 1.0 formatted files.
"""
- desc = list(stem.descriptor.parse_file(test.unit.descriptor.get_resource('bandwidth_file_v1.0'), 'badnwidth-file 1.0'))[0]
+ desc = list(stem.descriptor.parse_file(get_resource('bandwidth_file_v1.0'), 'badnwidth-file 1.0'))[0]
self.assertEqual(datetime.datetime(2019, 1, 14, 17, 41, 29), desc.timestamp)
self.assertEqual('1.0.0', desc.version)
@@ -41,7 +54,7 @@ class TestBandwidthFile(unittest.TestCase):
Parse version 1.2 formatted files.
"""
- desc = list(stem.descriptor.parse_file(test.unit.descriptor.get_resource('bandwidth_file_v1.2'), 'badnwidth-file 1.2'))[0]
+ desc = list(stem.descriptor.parse_file(get_resource('bandwidth_file_v1.2'), 'badnwidth-file 1.2'))[0]
self.assertEqual(datetime.datetime(2019, 1, 14, 5, 34, 59), desc.timestamp)
self.assertEqual('1.2.0', desc.version)
@@ -59,3 +72,80 @@ class TestBandwidthFile(unittest.TestCase):
self.assertEqual(96, desc.eligible_percent)
self.assertEqual(3908, desc.min_count)
self.assertEqual(60, desc.min_percent)
+
+ @patch('time.time', Mock(return_value = 1410723598.276578))
+ def test_minimal_bandwidth_file(self):
+ """
+ Basic sanity check that we can parse a bandwidth file with minimal
+ attributes.
+ """
+
+ desc = BandwidthFile.create()
+
+ self.assertEqual('1410723598', str(desc))
+
+ self.assertEqual(datetime.datetime(2014, 9, 14, 19, 39, 58), desc.timestamp)
+ self.assertEqual('1.0.0', desc.version)
+
+ self.assertEqual(None, desc.software)
+ self.assertEqual(None, desc.software_version)
+
+ self.assertEqual(None, desc.earliest_bandwidth)
+ self.assertEqual(None, desc.latest_bandwidth)
+ self.assertEqual(None, desc.created_at)
+ self.assertEqual(None, desc.generated_at)
+
+ self.assertEqual(None, desc.consensus_size)
+ self.assertEqual(None, desc.eligible_count)
+ self.assertEqual(None, desc.eligible_percent)
+ self.assertEqual(None, desc.min_count)
+ self.assertEqual(None, desc.min_percent)
+
+ self.assertEqual({}, desc.header)
+
+ def test_content_example(self):
+ """
+ Exercise the example in our content method's pydoc.
+ """
+
+ content = BandwidthFile.content({
+ 'timestamp': '12345',
+ 'version': '1.2.0',
+ 'content': [],
+ })
+
+ self.assertEqual('12345\nversion=1.2.0\n=====', content)
+
+ @patch('time.time', Mock(return_value = 1410723598.276578))
+ def test_new_header_attribute(self):
+ """
+ Include an unrecognized header field.
+ """
+
+ desc = BandwidthFile.create({'version': '1.1.0', 'new_header': 'neat stuff'})
+ self.assertEqual(EXPECTED_NEW_HEADER_CONTENT, str(desc))
+ self.assertEqual('1.1.0', desc.version)
+ self.assertEqual({'version': '1.1.0', 'new_header': 'neat stuff'}, desc.header)
+
+ def test_header_for_v1(self):
+ """
+ Document version 1.0 predates headers, and as such should be prohibited.
+ """
+
+ self.assertRaisesWith(ValueError, 'Headers require BandwidthFile version 1.1 or later', BandwidthFile.create, {'new_header': 'neat stuff'})
+
+ def test_invalid_timestamp(self):
+ """
+ Invalid timestamp values.
+ """
+
+ test_values = (
+ '',
+ 'boo',
+ '123.4',
+ '-123',
+ )
+
+ for value in test_values:
+ expected_exc = "First line should be a unix timestamp, but was '%s'" % value
+ self.assertRaisesWith(ValueError, expected_exc, BandwidthFile.create, {'timestamp': value})