Implemented composite design pattern
This commit is contained in:
@@ -1,5 +1,3 @@
|
|||||||
|
|
||||||
|
|
||||||
class Composite(object):
|
class Composite(object):
|
||||||
"""
|
"""
|
||||||
Composite class as part of the Composite pattern.
|
Composite class as part of the Composite pattern.
|
||||||
@@ -10,7 +8,6 @@ class Composite(object):
|
|||||||
"""
|
"""
|
||||||
self.components = set()
|
self.components = set()
|
||||||
self.interface = interface
|
self.interface = interface
|
||||||
self._method_names = [method for method in dir(interface) if callable(getattr(interface, method))]
|
|
||||||
|
|
||||||
def add_component(self, component):
|
def add_component(self, component):
|
||||||
"""
|
"""
|
||||||
@@ -18,14 +15,19 @@ class Composite(object):
|
|||||||
|
|
||||||
@param component: The component to add to this Composite
|
@param component: The component to add to this Composite
|
||||||
"""
|
"""
|
||||||
component_methods = [method for method in dir(component) if callable(getattr(component, method))]
|
valid = False
|
||||||
for method_name in self._method_names:
|
try:
|
||||||
if method_name in component_methods:
|
if component.interface == self.interface:
|
||||||
continue
|
valid = True
|
||||||
else:
|
except AttributeError:
|
||||||
raise AttributeError
|
if self.interface in component.__class__.__mro__:
|
||||||
else:
|
valid = True
|
||||||
|
finally:
|
||||||
|
if valid:
|
||||||
self.components.add(component)
|
self.components.add(component)
|
||||||
|
else:
|
||||||
|
raise AttributeError('Component {0} does not follow this composites interface {1}'.format(
|
||||||
|
component.__class__, self.interface))
|
||||||
|
|
||||||
def remove_component(self, component):
|
def remove_component(self, component):
|
||||||
"""
|
"""
|
||||||
|
|||||||
@@ -17,15 +17,7 @@ class CompositeTestCase(TestCase):
|
|||||||
def do_something(self):
|
def do_something(self):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
class LeafOne(Component):
|
class Leaf(Component):
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
self.did_something = False
|
|
||||||
|
|
||||||
def do_something(self):
|
|
||||||
self.did_something = True
|
|
||||||
|
|
||||||
class LeafTwo(Component):
|
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
self.did_something = False
|
self.did_something = False
|
||||||
@@ -34,50 +26,52 @@ class CompositeTestCase(TestCase):
|
|||||||
self.did_something = True
|
self.did_something = True
|
||||||
|
|
||||||
self.component_class = Component
|
self.component_class = Component
|
||||||
self.leaf_one = LeafOne()
|
self.leaf_one = Leaf()
|
||||||
self.leaf_two = LeafTwo()
|
self.leaf_two = Leaf()
|
||||||
|
self.leaf_three = Leaf()
|
||||||
def test_init(self):
|
|
||||||
|
|
||||||
composite = Composite(self.component_class)
|
|
||||||
|
|
||||||
self.assertIn('do_something', composite._method_names)
|
|
||||||
|
|
||||||
def test_add_invalid_component(self):
|
|
||||||
|
|
||||||
composite = Composite(self.component_class)
|
|
||||||
|
|
||||||
class BadComponent(object):
|
|
||||||
|
|
||||||
def foo(self):
|
|
||||||
pass
|
|
||||||
|
|
||||||
with self.assertRaises(AttributeError):
|
|
||||||
composite.add_component(BadComponent)
|
|
||||||
|
|
||||||
def test_add_component(self):
|
def test_add_component(self):
|
||||||
|
|
||||||
composite = Composite(self.component_class)
|
composite = Composite(self.component_class)
|
||||||
composite.add_component(self.leaf_one)
|
composite.add_component(self.leaf_one)
|
||||||
composite.add_component(self.leaf_two)
|
|
||||||
|
composite_two = Composite(self.component_class)
|
||||||
|
composite_two.add_component(self.leaf_two)
|
||||||
|
|
||||||
|
composite_three = Composite(self.component_class)
|
||||||
|
composite_three.add_component(self.leaf_three)
|
||||||
|
|
||||||
|
composite_two.add_component(composite_three)
|
||||||
|
|
||||||
|
composite.add_component(composite_two)
|
||||||
|
|
||||||
try:
|
try:
|
||||||
composite.add_component(self.leaf_two)
|
composite.add_component(composite_two)
|
||||||
|
composite_two.add_component(composite_three)
|
||||||
except:
|
except:
|
||||||
raise AssertionError
|
raise AssertionError()
|
||||||
else:
|
else:
|
||||||
self.assertSetEqual({self.leaf_one, self.leaf_two}, composite.components)
|
self.assertSetEqual({self.leaf_one, composite_two}, composite.components)
|
||||||
|
self.assertSetEqual({self.leaf_two, composite_three}, composite_two.components)
|
||||||
|
self.assertSetEqual({self.leaf_three}, composite_three.components)
|
||||||
|
|
||||||
def test_remove_component(self):
|
def test_remove_component(self):
|
||||||
|
|
||||||
composite = Composite(self.component_class)
|
composite = Composite(self.component_class)
|
||||||
|
|
||||||
|
composite_two = Composite(self.component_class)
|
||||||
|
composite_two.add_component(self.leaf_one)
|
||||||
|
composite_two.add_component(self.leaf_two)
|
||||||
|
|
||||||
composite.add_component(self.leaf_one)
|
composite.add_component(self.leaf_one)
|
||||||
composite.add_component(self.leaf_two)
|
composite.add_component(self.leaf_two)
|
||||||
|
composite.add_component(composite_two)
|
||||||
|
|
||||||
composite.remove_component(self.leaf_one)
|
composite.remove_component(self.leaf_one)
|
||||||
composite.remove_component(self.leaf_two)
|
composite.remove_component(self.leaf_two)
|
||||||
|
composite.remove_component(composite_two)
|
||||||
try:
|
try:
|
||||||
composite.remove_component(self.leaf_two)
|
composite.remove_component(composite_two)
|
||||||
except:
|
except:
|
||||||
raise AssertionError
|
raise AssertionError
|
||||||
else:
|
else:
|
||||||
@@ -86,40 +80,81 @@ class CompositeTestCase(TestCase):
|
|||||||
def test_delegate(self):
|
def test_delegate(self):
|
||||||
|
|
||||||
composite = Composite(self.component_class)
|
composite = Composite(self.component_class)
|
||||||
|
composite_two = Composite(self.component_class)
|
||||||
|
composite_three = Composite(self.component_class)
|
||||||
|
|
||||||
composite.add_component(self.leaf_one)
|
composite.add_component(self.leaf_one)
|
||||||
composite.add_component(self.leaf_two)
|
composite_two.add_component(self.leaf_two)
|
||||||
|
composite_three.add_component(self.leaf_three)
|
||||||
|
|
||||||
|
composite_two.add_component(composite_three)
|
||||||
|
composite.add_component(composite_two)
|
||||||
|
|
||||||
composite.delegate('do_something')
|
composite.delegate('do_something')
|
||||||
|
|
||||||
self.assertTrue(self.leaf_one.did_something)
|
self.assertTrue(self.leaf_one.did_something)
|
||||||
self.assertTrue(self.leaf_two.did_something)
|
self.assertTrue(self.leaf_two.did_something)
|
||||||
|
self.assertTrue(self.leaf_three.did_something)
|
||||||
|
|
||||||
self.leaf_one.did_something = False
|
self.leaf_one.did_something = False
|
||||||
self.leaf_two.did_something = False
|
self.leaf_two.did_something = False
|
||||||
|
self.leaf_three.did_something = False
|
||||||
|
|
||||||
def test_getattr(self):
|
def test_getattr(self):
|
||||||
|
|
||||||
composite = Composite(self.component_class)
|
composite = Composite(self.component_class)
|
||||||
|
composite_two = Composite(self.component_class)
|
||||||
|
composite_three = Composite(self.component_class)
|
||||||
|
|
||||||
composite.add_component(self.leaf_one)
|
composite.add_component(self.leaf_one)
|
||||||
composite.add_component(self.leaf_two)
|
composite_two.add_component(self.leaf_two)
|
||||||
|
composite_three.add_component(self.leaf_three)
|
||||||
|
|
||||||
|
composite_two.add_component(composite_three)
|
||||||
|
composite.add_component(composite_two)
|
||||||
|
|
||||||
composite.do_something()
|
composite.do_something()
|
||||||
|
|
||||||
self.assertTrue(self.leaf_one.did_something)
|
self.assertTrue(self.leaf_one.did_something)
|
||||||
self.assertTrue(self.leaf_two.did_something)
|
self.assertTrue(self.leaf_two.did_something)
|
||||||
|
self.assertTrue(self.leaf_three.did_something)
|
||||||
|
|
||||||
self.leaf_one.did_something = False
|
self.leaf_one.did_something = False
|
||||||
self.leaf_two.did_something = False
|
self.leaf_two.did_something = False
|
||||||
|
self.leaf_three.did_something = False
|
||||||
|
|
||||||
def test_invalid_getattr(self):
|
def test_invalid_getattr(self):
|
||||||
|
|
||||||
composite = Composite(self.component_class)
|
composite = Composite(self.component_class)
|
||||||
|
composite_two = Composite(self.component_class)
|
||||||
|
composite_three = Composite(self.component_class)
|
||||||
|
|
||||||
composite.add_component(self.leaf_one)
|
composite.add_component(self.leaf_one)
|
||||||
composite.add_component(self.leaf_two)
|
composite_two.add_component(self.leaf_two)
|
||||||
|
composite_three.add_component(self.leaf_three)
|
||||||
|
|
||||||
|
composite_two.add_component(composite_three)
|
||||||
|
composite.add_component(composite_two)
|
||||||
|
|
||||||
with self.assertRaises(AttributeError):
|
with self.assertRaises(AttributeError):
|
||||||
composite.foo()
|
composite.foo()
|
||||||
composite.did_something()
|
composite.did_something()
|
||||||
|
|
||||||
|
def test_interface(self):
|
||||||
|
|
||||||
|
class BadComponent(object):
|
||||||
|
def foo(self):
|
||||||
|
raise NotImplementedError()
|
||||||
|
|
||||||
|
class BadLeaf(BadComponent):
|
||||||
|
def foo(self):
|
||||||
|
pass
|
||||||
|
|
||||||
|
composite = Composite(self.component_class)
|
||||||
|
composite_two = Composite(BadComponent)
|
||||||
|
composite_two.add_component(BadLeaf())
|
||||||
|
|
||||||
|
self.assertRaises(AttributeError, composite_two.add_component, self.leaf_one)
|
||||||
|
self.assertRaises(AttributeError, composite.add_component, composite_two)
|
||||||
|
self.assertRaises(AttributeError, composite.add_component, BadLeaf())
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user