Passed
Pull Request — dev (#30)
by Konstantinos
03:11 queued 01:42
created

test_metaclass_usage()   A

Complexity

Conditions 1

Size

Total Lines 5
Code Lines 4

Duplication

Lines 0
Ratio 0 %

Importance

Changes 0
Metric Value
cc 1
eloc 4
nop 2
dl 0
loc 5
rs 10
c 0
b 0
f 0
1
import pytest
2
3
4
@pytest.fixture
5
def subclass_registry_metaclass():
6
    from so_magic.utils import SubclassRegistry
7
    return SubclassRegistry
8
9
10
@pytest.fixture
11
def assert_correct_class_creation(subclass_registry_metaclass):
12
    def assert_class_creation(user_class):
13
        assert type(user_class) == subclass_registry_metaclass
14
        assert hasattr(user_class, 'subclasses')
15
        assert hasattr(user_class, 'create')
16
        assert hasattr(user_class, 'register_as_subclass')
17
        assert user_class.subclasses == {}
18
    return assert_class_creation
19
20
21
@pytest.fixture
22
def register_class(subclass_registry_metaclass):
23
    def _register_class(subclass_id: str, inherit=False):
24
        class ParentClass(metaclass=subclass_registry_metaclass):
25
            pass
26
27
        if inherit:
28
            @ParentClass.register_as_subclass(subclass_id)
29
            class Child(ParentClass):
30
                pass
31
        else:
32
            @ParentClass.register_as_subclass(subclass_id)
33
            class Child:
34
                pass
35
36
        child_instance = ParentClass.create(subclass_id)
37
38
        return {'class_registry': ParentClass, 'child': Child, 'child_instance': child_instance}
39
    return _register_class
40
41
42
@pytest.fixture
43
def usage_with_subclass(register_class, assert_correct_metaclass_behaviour):
44
    def parent_n_child_classes(subclass_id: str):
45
        classes = register_class(subclass_id, inherit=True)
46
47
        assert_correct_metaclass_behaviour(classes, subclass_id)
48
        assert isinstance(classes['child_instance'], classes['class_registry'])
49
50
        return classes['child_instance'], classes['child'], classes['class_registry']
51
    return parent_n_child_classes
52
53
54
@pytest.fixture
55
def plain_usage(register_class, assert_correct_metaclass_behaviour):
56
    def parent_n_child_classes(subclass_id: str):
57
        classes = register_class(subclass_id, inherit=False)
58
59
        assert_correct_metaclass_behaviour(classes, subclass_id)
60
        assert not isinstance(classes['child_instance'], classes['class_registry'])
61
62
        return classes['child_instance'], classes['child'], classes['class_registry']
63
    return parent_n_child_classes
64
65
66
@pytest.fixture
67
def assert_correct_metaclass_behaviour():
68
    def assert_metaclass_behaviour(classes, subclass_id):
69
        assert classes['class_registry'].subclasses[subclass_id] == classes['child']
70
        assert type(classes['child_instance']) == classes['child']
71
        assert isinstance(classes['child_instance'], classes['child'])
72
    return assert_metaclass_behaviour
73
74
75
def test_metaclass_usage(subclass_registry_metaclass, assert_correct_class_creation):
76
    class ParentClass(metaclass=subclass_registry_metaclass):
77
        pass
78
79
    assert_correct_class_creation(ParentClass)
80
81
82
def test_subclass_registry(usage_with_subclass, plain_usage):
83
    child1_instance1, Child1, ParentClass = usage_with_subclass('child1')
84
85
    non_existent_identifier = 'child2'
86
87
    exception_message_regex = \
88
        f'Bad "{str(ParentClass.__name__)}" subclass request; requested subclass with identifier ' \
89
        f'{non_existent_identifier}, but known identifiers are ' \
90
        rf'\[{", ".join(subclass_identifier for subclass_identifier in ParentClass.subclasses.keys())}\]'
91
92
    with pytest.raises(ValueError, match=exception_message_regex):
93
        ParentClass.create(non_existent_identifier)
94
95
    child1_instance2, Child2, ParentClass2 = plain_usage('child2')
96
    assert ParentClass.subclasses['child1'] == Child1
97