Implemented Singleton metaclass

This commit is contained in:
tylerlaberge
2016-07-17 21:29:52 -04:00
parent 8e26a8b0d8
commit 13eac458e5
4 changed files with 91 additions and 32 deletions

View File

@@ -1,4 +1,19 @@
class Singleton(object): class Singleton(type):
pass """
Singleton Metaclass.
Enforces any object using this metaclass to only create a single instance.
"""
__instance = None
def __call__(cls, *args, **kwargs):
"""
Override the __call__ method to make sure only one instance is created.
"""
if cls.__instance is None:
cls.__instance = type.__call__(cls, *args, **kwargs)
return cls.__instance

View File

@@ -1,6 +1,6 @@
from unittest import TestCase from unittest import TestCase
from tests.utils.dummy_class import dummy_factory from tests.utils.dummy import dummy_class_factory
from patterns.creational import Singleton from patterns.creational import Singleton
@@ -12,15 +12,40 @@ class SingletonTestCase(TestCase):
""" """
Initialize testing data. Initialize testing data.
""" """
self.dummy_class = dummy_factory(base_class=Singleton, attributes={}, functions={}) self.dummy_class_one = dummy_class_factory(meta_class=Singleton,
attributes={},
functions={})
def test_id(self): self.dummy_class_two = dummy_class_factory(meta_class=Singleton,
attributes={},
functions={})
def test_single(self):
""" """
Test the id's of two singleton instances. Test instances from a single singleton class.
@raise AssertionError: If the test fails. @raise AssertionError: If the test fails.
""" """
dummy = self.dummy_class() dummy_one = self.dummy_class_one()
dummy_2 = self.dummy_class() dummy_two = self.dummy_class_one()
self.assertEquals(id(dummy_one), id(dummy_two))
def test_multiple(self):
"""
Test instances from multiple singleton classes.
@raise AssertionError: If the test fails.
"""
dummy_class_one_instance_one = self.dummy_class_one()
dummy_class_one_instance_two = self.dummy_class_one()
dummy_class_two_instance_one = self.dummy_class_two()
dummy_class_two_instance_two = self.dummy_class_two()
self.assertEquals(id(dummy_class_one_instance_one), id(dummy_class_one_instance_two))
self.assertEquals(id(dummy_class_two_instance_one), id(dummy_class_two_instance_two))
self.assertNotEquals(id(dummy_class_one_instance_one), id(dummy_class_two_instance_one))
self.assertNotEquals(id(dummy_class_one_instance_two), id(dummy_class_two_instance_two))
self.assertEquals(id(dummy), id(dummy_2))

View File

@@ -1,5 +1,6 @@
from unittest import TestCase from unittest import TestCase
from tests.utils.dummy_class import dummy_factory from tests.utils.dummy import dummy_class_factory
from abc import ABCMeta, ABC
class DummyClassTestCase(TestCase): class DummyClassTestCase(TestCase):
@@ -21,9 +22,8 @@ class DummyClassTestCase(TestCase):
@raise AssertionError: If the test fails. @raise AssertionError: If the test fails.
""" """
dummy_class = dummy_factory(base_class=object, dummy_class = dummy_class_factory(attributes={'a': self.a, 'b': self.b},
attributes={'a': self.a, 'b': self.b}, functions={'add': self.add_function, 'subtract': self.subtract_function})
functions={'add': self.add_function, 'subtract': self.subtract_function})
dummy = dummy_class() dummy = dummy_class()
@@ -36,9 +36,8 @@ class DummyClassTestCase(TestCase):
@raise AssertionError: If the test fails. @raise AssertionError: If the test fails.
""" """
dummy_class = dummy_factory(base_class=object, dummy_class = dummy_class_factory(attributes={'a': self.a, 'b': self.b},
attributes={'a': self.a, 'b': self.b}, functions={'add': self.add_function})
functions={'add': self.add_function})
dummy = dummy_class() dummy = dummy_class()
self.assertEquals(15, dummy.add()) self.assertEquals(15, dummy.add())
@@ -49,9 +48,8 @@ class DummyClassTestCase(TestCase):
@raise AssertionError: If the test fails. @raise AssertionError: If the test fails.
""" """
dummy_class = dummy_factory(base_class=object, dummy_class = dummy_class_factory(attributes={'a': self.a, 'b': self.b},
attributes={'a': self.a, 'b': self.b}, functions={'subtract': self.subtract_function})
functions={'subtract': self.subtract_function})
dummy = dummy_class() dummy = dummy_class()
self.assertEquals(5, dummy.subtract()) self.assertEquals(5, dummy.subtract())
@@ -62,9 +60,8 @@ class DummyClassTestCase(TestCase):
@raise AssertionError: If the test fails. @raise AssertionError: If the test fails.
""" """
dummy_class_one = dummy_factory(base_class=object, dummy_class_one = dummy_class_factory(attributes={'a': self.a, 'b': self.b},
attributes={'a': self.a, 'b': self.b}, functions={'add': self.add_function})
functions={'add': self.add_function})
dummy_class_one_instance_one = dummy_class_one() dummy_class_one_instance_one = dummy_class_one()
dummy_class_one_instance_two = dummy_class_one() dummy_class_one_instance_two = dummy_class_one()
@@ -79,12 +76,10 @@ class DummyClassTestCase(TestCase):
@raise AssertionError: If the test fails. @raise AssertionError: If the test fails.
""" """
dummy_class_one = dummy_factory(base_class=object, dummy_class_one = dummy_class_factory(attributes={'a': self.a, 'b': self.b},
attributes={'a': self.a, 'b': self.b}, functions={'add': self.add_function})
functions={'add': self.add_function}) dummy_class_two = dummy_class_factory(attributes={'a': 30, 'b': 10},
dummy_class_two = dummy_factory(base_class=object, functions={'subtract': self.subtract_function})
attributes={'a': 30, 'b': 10},
functions={'subtract': self.subtract_function})
self.assertNotEquals(dummy_class_one.a, dummy_class_two.a) self.assertNotEquals(dummy_class_one.a, dummy_class_two.a)
self.assertNotEquals(dummy_class_one.b, dummy_class_two.b) self.assertNotEquals(dummy_class_one.b, dummy_class_two.b)
@@ -95,6 +90,29 @@ class DummyClassTestCase(TestCase):
assert (hasattr(dummy_class_two, 'subtract')) assert (hasattr(dummy_class_two, 'subtract'))
assert (not hasattr(dummy_class_two, 'add')) assert (not hasattr(dummy_class_two, 'add'))
def test_meta_class(self):
"""
Test assigning a metaclass.
@raise AssertionError: If the test fails.
"""
dummy_class = dummy_class_factory(attributes={'a': self.a, 'b': self.b},
functions={'add': self.add_function},
meta_class=ABCMeta)
self.assertEquals(ABCMeta, dummy_class.__class__)
def test_base_class(self):
"""
Test assigning a base class.
@raise AssertionError: If the test fails
"""
dummy_class = dummy_class_factory(attributes={'a': self.a, 'b': self.b},
functions={'add': self.add_function},
base_class=ABC)
self.assertEquals(ABC, dummy_class.__base__)

View File

@@ -1,13 +1,14 @@
from types import MethodType from types import MethodType
def dummy_factory(base_class, attributes, functions): def dummy_class_factory(attributes, functions, base_class=object, meta_class=type):
class DummyClass(base_class): class DummyClass(base_class, metaclass=meta_class):
""" """
Class representing dummy data. Class representing dummy data.
""" """
pass def __init__(self):
pass
for key, value in attributes.items(): for key, value in attributes.items():
if callable(value): if callable(value):
@@ -21,4 +22,4 @@ def dummy_factory(base_class, attributes, functions):
else: else:
setattr(DummyClass, key, MethodType(value, DummyClass)) setattr(DummyClass, key, MethodType(value, DummyClass))
return DummyClass return DummyClass