File: types.py

package info (click to toggle)
blueprint-compiler 0.18.0-2
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid
  • size: 2,140 kB
  • sloc: python: 8,504; sh: 31; makefile: 6
file content (184 lines) | stat: -rw-r--r-- 6,219 bytes parent folder | download | duplicates (2)
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
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
# types.py
#
# Copyright 2022 James Westman <james@jwestman.net>
#
# This file 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 3 of the
# License, or (at your option) any later version.
#
# This file 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 program.  If not, see <http://www.gnu.org/licenses/>.
#
# SPDX-License-Identifier: LGPL-3.0-or-later


from ..gir import Class, ExternType, Interface
from .common import *


class TypeName(AstNode):
    grammar = AnyOf(
        [
            UseIdent("namespace"),
            ".",
            UseIdent("class_name"),
        ],
        [
            AnyOf("$", [".", UseLiteral("old_extern", True)]),
            UseIdent("class_name"),
            UseLiteral("extern", True),
        ],
        UseIdent("class_name"),
    )

    @validate()
    def old_extern(self):
        if self.tokens["old_extern"]:
            raise UpgradeWarning(
                "Use the '$' extern syntax introduced in blueprint 0.8.0",
                actions=[CodeAction("Use '$' syntax", "$" + self.tokens["class_name"])],
            )

    @validate("class_name")
    def type_exists(self):
        if not self.tokens["extern"] and self.gir_ns is not None:
            self.root.gir.validate_type(
                self.tokens["class_name"], self.tokens["namespace"]
            )

    @validate("namespace")
    def gir_ns_exists(self):
        if not self.tokens["extern"]:
            try:
                self.root.gir.validate_ns(self.tokens["namespace"])
            except CompileError as e:
                ns = self.tokens["namespace"]
                e.actions = [
                    self.root.import_code_action(n, version)
                    for n, version in gir.get_available_namespaces()
                    if n == ns
                ]
                raise e

    @validate()
    def deprecated(self) -> None:
        if self.gir_type and self.gir_type.deprecated:
            hints = []
            if self.gir_type.deprecated_doc:
                hints.append(self.gir_type.deprecated_doc)
            raise DeprecatedWarning(
                f"{self.gir_type.full_name} is deprecated",
                hints=hints,
            )

    @property
    def gir_ns(self) -> T.Optional[gir.Namespace]:
        if not self.tokens["extern"]:
            return self.root.gir.namespaces.get(self.tokens["namespace"] or "Gtk")
        return None

    @property
    def gir_type(self) -> gir.GirType:
        if self.tokens["class_name"] and not self.tokens["extern"]:
            return self.root.gir.get_type(
                self.tokens["class_name"], self.tokens["namespace"]
            )

        return gir.ExternType(self.tokens["class_name"])

    @property
    def glib_type_name(self) -> str:
        if gir_type := self.gir_type:
            return gir_type.glib_type_name
        else:
            return self.tokens["class_name"]

    @docs("namespace")
    def namespace_docs(self):
        if ns := self.root.gir.namespaces.get(self.tokens["namespace"]):
            return ns.doc

    @docs("class_name")
    def class_docs(self):
        if self.gir_type:
            return self.gir_type.doc

    @property
    def as_string(self) -> str:
        if self.tokens["extern"]:
            return "$" + self.tokens["class_name"]
        elif self.tokens["namespace"]:
            return f"{self.tokens['namespace']}.{self.tokens['class_name']}"
        else:
            return self.tokens["class_name"]


class ClassName(TypeName):
    @validate("namespace", "class_name")
    def gir_class_exists(self):
        if (
            self.gir_type is not None
            and not isinstance(self.gir_type, ExternType)
            and not isinstance(self.gir_type, Class)
        ):
            if isinstance(self.gir_type, Interface):
                raise CompileError(
                    f"{self.gir_type.full_name} is an interface, not a class"
                )
            else:
                raise CompileError(f"{self.gir_type.full_name} is not a class")


class ConcreteClassName(ClassName):
    @validate("namespace", "class_name")
    def not_abstract(self):
        if isinstance(self.gir_type, Class) and self.gir_type.abstract:
            raise CompileError(
                f"{self.gir_type.full_name} can't be instantiated because it's abstract",
                hints=[f"did you mean to use a subclass of {self.gir_type.full_name}?"],
            )


class TemplateClassName(ClassName):
    """Handles the special case of a template type. The old syntax uses an identifier,
    which is ambiguous with the new syntax. So this class displays an appropriate
    upgrade warning instead of a class not found error."""

    @property
    def is_legacy(self):
        return (
            self.tokens["extern"] is None
            and self.tokens["namespace"] is None
            and self.root.gir.get_type(self.tokens["class_name"], "Gtk") is None
        )

    @property
    def gir_type(self) -> gir.GirType:
        if self.is_legacy:
            return gir.ExternType(self.tokens["class_name"])
        else:
            return super().gir_type

    @validate("class_name")
    def type_exists(self):
        if self.is_legacy:
            if type := self.root.gir.get_type_by_cname(self.tokens["class_name"]):
                replacement = type.full_name
            else:
                replacement = "$" + self.tokens["class_name"]

            raise UpgradeWarning(
                "Use type syntax here (introduced in blueprint 0.8.0)",
                actions=[CodeAction("Use type syntax", replace_with=replacement)],
            )

        if not self.tokens["extern"] and self.gir_ns is not None:
            self.root.gir.validate_type(
                self.tokens["class_name"], self.tokens["namespace"]
            )