summaryrefslogtreecommitdiff
path: root/src/buildstream/_types.pyx
blob: 0f8d3690cb606e32eead2b9826dfabf9d70da64b (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
#
#  Copyright (C) 2018 Bloomberg LP
#
#  This program is free software; you can redistribute it and/or
#  modify it under the terms of the GNU Lesser General Public
#  License as published by the Free Software Foundation; either
#  version 2 of the License, or (at your option) any later version.
#
#  This library is distributed in the hope that it will be useful,
#  but WITHOUT ANY WARRANTY; without even the implied warranty of
#  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.	 See the GNU
#  Lesser General Public License for more details.
#
#  You should have received a copy of the GNU Lesser General Public
#  License along with this library. If not, see <http://www.gnu.org/licenses/>.
#
#  Authors:
#        Tristan Van Berkom <tristan.vanberkom@codethink.co.uk>
#        Jim MacArthur <jim.macarthur@codethink.co.uk>
#        Benjamin Schubert <bschubert15@bloomberg.net>

# MetaFastEnum()
#
# This is a reemplementation of MetaEnum, in order to get a faster implementation of Enum.
#
# Enum turns out to be very slow and would add a noticeable slowdown when we try to add them to the codebase.
# We therefore reimplement a subset of the `Enum` functionality that keeps it compatible with normal `Enum`.
# That way, any place in the code base that access a `FastEnum`, can normally also accept an `Enum`. The reverse
# is not correct, since we only implement a subset of `Enum`.
class MetaFastEnum(type):
    def __new__(mcs, name, bases, dct):
        if name == "FastEnum":
            return type.__new__(mcs, name, bases, dct)

        assert len(bases) == 1, "Multiple inheritance with Fast enums is not currently supported."

        dunder_values = {}
        normal_values = {}

        parent_keys = bases[0].__dict__.keys()

        assert "__class__" not in dct.keys(), "Overriding '__class__' is not allowed on 'FastEnum' classes"

        for key, value in dct.items():
            if key.startswith("__") and key.endswith("__"):
                dunder_values[key] = value
            else:
                assert key not in parent_keys, "Overriding 'FastEnum.{}' is not allowed. ".format(key)
                normal_values[key] = value

        kls = type.__new__(mcs, name, bases, dunder_values)
        mcs.set_values(kls, normal_values)

        return kls

    @classmethod
    def set_values(mcs, kls, data):
        value_to_entry = {}

        assert len(set(data.values())) == len(data.values()), "Values for {} are not unique".format(kls)
        assert len(set(type(value) for value in data.values())) <= 1, \
            "Values of {} are of heterogeneous types".format(kls)

        for key, value in data.items():
            new_value = object.__new__(kls)
            object.__setattr__(new_value, "value", value)
            object.__setattr__(new_value, "name", key)

            type.__setattr__(kls, key, new_value)

            value_to_entry[value] = new_value

        type.__setattr__(kls, "_value_to_entry", value_to_entry)

    def __repr__(self):
        return "<fastenum '{}'>".format(self.__name__)

    def __setattr__(self, key, value):
        raise ValueError("Adding new values dynamically is not supported")

    def __iter__(self):
        return iter(self._value_to_entry.values())