Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1 | # -*- coding: utf-8 -*- |
| 2 | # |
| 3 | # QAPI schema internal representation |
| 4 | # |
| 5 | # Copyright (c) 2015-2019 Red Hat Inc. |
| 6 | # |
| 7 | # Authors: |
| 8 | # Markus Armbruster <armbru@redhat.com> |
| 9 | # Eric Blake <eblake@redhat.com> |
| 10 | # Marc-André Lureau <marcandre.lureau@redhat.com> |
| 11 | # |
| 12 | # This work is licensed under the terms of the GNU GPL, version 2. |
| 13 | # See the COPYING file in the top-level directory. |
| 14 | |
John Snow | ce7fde0 | 2024-03-15 16:22:40 +0100 | [diff] [blame] | 15 | # pylint: disable=too-many-lines |
| 16 | |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 17 | # TODO catching name collisions in generated code would be nice |
| 18 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 19 | from __future__ import annotations |
| 20 | |
John Snow | d150be3 | 2024-03-15 16:22:44 +0100 | [diff] [blame] | 21 | from abc import ABC, abstractmethod |
John Snow | 67fea57 | 2020-10-09 12:15:29 -0400 | [diff] [blame] | 22 | from collections import OrderedDict |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 23 | import os |
| 24 | import re |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 25 | from typing import ( |
| 26 | Any, |
| 27 | Callable, |
| 28 | Dict, |
| 29 | List, |
| 30 | Optional, |
| 31 | Union, |
| 32 | cast, |
| 33 | ) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 34 | |
Marc-André Lureau | d806f89 | 2021-08-04 12:31:00 +0400 | [diff] [blame] | 35 | from .common import ( |
| 36 | POINTER_SUFFIX, |
| 37 | c_name, |
| 38 | cgen_ifcond, |
| 39 | docgen_ifcond, |
Markus Armbruster | 1889e57 | 2021-08-31 14:37:58 +0200 | [diff] [blame] | 40 | gen_endif, |
| 41 | gen_if, |
Marc-André Lureau | d806f89 | 2021-08-04 12:31:00 +0400 | [diff] [blame] | 42 | ) |
John Snow | 3404e57 | 2021-05-19 14:39:37 -0400 | [diff] [blame] | 43 | from .error import QAPIError, QAPISemError, QAPISourceError |
John Snow | 7137a96 | 2020-10-09 12:15:27 -0400 | [diff] [blame] | 44 | from .expr import check_exprs |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 45 | from .parser import QAPIDoc, QAPIExpression, QAPISchemaParser |
| 46 | from .source import QAPISourceInfo |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 47 | |
| 48 | |
Marc-André Lureau | f17539c | 2021-08-04 12:30:57 +0400 | [diff] [blame] | 49 | class QAPISchemaIfCond: |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 50 | def __init__( |
| 51 | self, |
| 52 | ifcond: Optional[Union[str, Dict[str, object]]] = None, |
| 53 | ) -> None: |
Markus Armbruster | e46c930 | 2021-08-31 14:37:59 +0200 | [diff] [blame] | 54 | self.ifcond = ifcond |
Marc-André Lureau | f17539c | 2021-08-04 12:30:57 +0400 | [diff] [blame] | 55 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 56 | def _cgen(self) -> str: |
Marc-André Lureau | 6cc2e48 | 2021-08-04 12:30:59 +0400 | [diff] [blame] | 57 | return cgen_ifcond(self.ifcond) |
| 58 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 59 | def gen_if(self) -> str: |
Markus Armbruster | 1889e57 | 2021-08-31 14:37:58 +0200 | [diff] [blame] | 60 | return gen_if(self._cgen()) |
| 61 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 62 | def gen_endif(self) -> str: |
Markus Armbruster | 1889e57 | 2021-08-31 14:37:58 +0200 | [diff] [blame] | 63 | return gen_endif(self._cgen()) |
| 64 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 65 | def docgen(self) -> str: |
Marc-André Lureau | d806f89 | 2021-08-04 12:31:00 +0400 | [diff] [blame] | 66 | return docgen_ifcond(self.ifcond) |
| 67 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 68 | def is_present(self) -> bool: |
Marc-André Lureau | 33aa326 | 2021-08-04 12:30:58 +0400 | [diff] [blame] | 69 | return bool(self.ifcond) |
| 70 | |
Marc-André Lureau | f17539c | 2021-08-04 12:30:57 +0400 | [diff] [blame] | 71 | |
Markus Armbruster | baa310f | 2020-03-04 16:59:29 +0100 | [diff] [blame] | 72 | class QAPISchemaEntity: |
John Snow | 2418d1c | 2024-03-15 16:22:41 +0100 | [diff] [blame] | 73 | """ |
| 74 | A schema entity. |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 75 | |
John Snow | 2418d1c | 2024-03-15 16:22:41 +0100 | [diff] [blame] | 76 | This is either a directive, such as include, or a definition. |
| 77 | The latter uses sub-class `QAPISchemaDefinition`. |
| 78 | """ |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 79 | def __init__(self, info: Optional[QAPISourceInfo]): |
| 80 | self._module: Optional[QAPISchemaModule] = None |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 81 | # For explicitly defined entities, info points to the (explicit) |
| 82 | # definition. For builtins (and their arrays), info is None. |
| 83 | # For implicitly defined entities, info points to a place that |
| 84 | # triggered the implicit definition (there may be more than one |
| 85 | # such place). |
| 86 | self.info = info |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 87 | self._checked = False |
| 88 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 89 | def __repr__(self) -> str: |
John Snow | 2418d1c | 2024-03-15 16:22:41 +0100 | [diff] [blame] | 90 | return "<%s at 0x%x>" % (type(self).__name__, id(self)) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 91 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 92 | def check(self, schema: QAPISchema) -> None: |
John Snow | ce7fde0 | 2024-03-15 16:22:40 +0100 | [diff] [blame] | 93 | # pylint: disable=unused-argument |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 94 | self._checked = True |
| 95 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 96 | def connect_doc(self, doc: Optional[QAPIDoc] = None) -> None: |
John Snow | 2418d1c | 2024-03-15 16:22:41 +0100 | [diff] [blame] | 97 | pass |
Markus Armbruster | ee1e6a1 | 2019-10-24 13:02:26 +0200 | [diff] [blame] | 98 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 99 | def _set_module( |
| 100 | self, schema: QAPISchema, info: Optional[QAPISourceInfo] |
| 101 | ) -> None: |
Markus Armbruster | a9f1dd7 | 2019-11-20 19:25:49 +0100 | [diff] [blame] | 102 | assert self._checked |
John Snow | 39b2d83 | 2021-02-01 14:37:41 -0500 | [diff] [blame] | 103 | fname = info.fname if info else QAPISchemaModule.BUILTIN_MODULE_NAME |
John Snow | e2bbc4e | 2021-02-01 14:37:39 -0500 | [diff] [blame] | 104 | self._module = schema.module_by_fname(fname) |
Markus Armbruster | 3e7fb58 | 2019-11-20 19:25:50 +0100 | [diff] [blame] | 105 | self._module.add_entity(self) |
Markus Armbruster | a9f1dd7 | 2019-11-20 19:25:49 +0100 | [diff] [blame] | 106 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 107 | def set_module(self, schema: QAPISchema) -> None: |
Markus Armbruster | a9f1dd7 | 2019-11-20 19:25:49 +0100 | [diff] [blame] | 108 | self._set_module(schema, self.info) |
| 109 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 110 | def visit(self, visitor: QAPISchemaVisitor) -> None: |
John Snow | 2418d1c | 2024-03-15 16:22:41 +0100 | [diff] [blame] | 111 | # pylint: disable=unused-argument |
| 112 | assert self._checked |
| 113 | |
| 114 | |
| 115 | class QAPISchemaDefinition(QAPISchemaEntity): |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 116 | meta: str |
John Snow | 2418d1c | 2024-03-15 16:22:41 +0100 | [diff] [blame] | 117 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 118 | def __init__( |
| 119 | self, |
| 120 | name: str, |
| 121 | info: Optional[QAPISourceInfo], |
| 122 | doc: Optional[QAPIDoc], |
| 123 | ifcond: Optional[QAPISchemaIfCond] = None, |
| 124 | features: Optional[List[QAPISchemaFeature]] = None, |
| 125 | ): |
John Snow | 2418d1c | 2024-03-15 16:22:41 +0100 | [diff] [blame] | 126 | super().__init__(info) |
| 127 | for f in features or []: |
John Snow | 2418d1c | 2024-03-15 16:22:41 +0100 | [diff] [blame] | 128 | f.set_defined_in(name) |
| 129 | self.name = name |
| 130 | self.doc = doc |
| 131 | self._ifcond = ifcond or QAPISchemaIfCond() |
| 132 | self.features = features or [] |
| 133 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 134 | def __repr__(self) -> str: |
John Snow | 2418d1c | 2024-03-15 16:22:41 +0100 | [diff] [blame] | 135 | return "<%s:%s at 0x%x>" % (type(self).__name__, self.name, |
| 136 | id(self)) |
| 137 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 138 | def c_name(self) -> str: |
John Snow | 2418d1c | 2024-03-15 16:22:41 +0100 | [diff] [blame] | 139 | return c_name(self.name) |
| 140 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 141 | def check(self, schema: QAPISchema) -> None: |
John Snow | 2418d1c | 2024-03-15 16:22:41 +0100 | [diff] [blame] | 142 | assert not self._checked |
| 143 | super().check(schema) |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 144 | seen: Dict[str, QAPISchemaMember] = {} |
John Snow | 2418d1c | 2024-03-15 16:22:41 +0100 | [diff] [blame] | 145 | for f in self.features: |
| 146 | f.check_clash(self.info, seen) |
| 147 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 148 | def connect_doc(self, doc: Optional[QAPIDoc] = None) -> None: |
John Snow | 2418d1c | 2024-03-15 16:22:41 +0100 | [diff] [blame] | 149 | super().connect_doc(doc) |
| 150 | doc = doc or self.doc |
| 151 | if doc: |
| 152 | for f in self.features: |
| 153 | doc.connect_feature(f) |
| 154 | |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 155 | @property |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 156 | def ifcond(self) -> QAPISchemaIfCond: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 157 | assert self._checked |
| 158 | return self._ifcond |
| 159 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 160 | def is_implicit(self) -> bool: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 161 | return not self.info |
| 162 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 163 | def describe(self) -> str: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 164 | return "%s '%s'" % (self.meta, self.name) |
| 165 | |
| 166 | |
Markus Armbruster | baa310f | 2020-03-04 16:59:29 +0100 | [diff] [blame] | 167 | class QAPISchemaVisitor: |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 168 | def visit_begin(self, schema: QAPISchema) -> None: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 169 | pass |
| 170 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 171 | def visit_end(self) -> None: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 172 | pass |
| 173 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 174 | def visit_module(self, name: str) -> None: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 175 | pass |
| 176 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 177 | def visit_needed(self, entity: QAPISchemaEntity) -> bool: |
John Snow | ce7fde0 | 2024-03-15 16:22:40 +0100 | [diff] [blame] | 178 | # pylint: disable=unused-argument |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 179 | # Default to visiting everything |
| 180 | return True |
| 181 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 182 | def visit_include(self, name: str, info: Optional[QAPISourceInfo]) -> None: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 183 | pass |
| 184 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 185 | def visit_builtin_type( |
| 186 | self, name: str, info: Optional[QAPISourceInfo], json_type: str |
| 187 | ) -> None: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 188 | pass |
| 189 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 190 | def visit_enum_type( |
| 191 | self, |
| 192 | name: str, |
| 193 | info: Optional[QAPISourceInfo], |
| 194 | ifcond: QAPISchemaIfCond, |
| 195 | features: List[QAPISchemaFeature], |
| 196 | members: List[QAPISchemaEnumMember], |
| 197 | prefix: Optional[str], |
| 198 | ) -> None: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 199 | pass |
| 200 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 201 | def visit_array_type( |
| 202 | self, |
| 203 | name: str, |
| 204 | info: Optional[QAPISourceInfo], |
| 205 | ifcond: QAPISchemaIfCond, |
| 206 | element_type: QAPISchemaType, |
| 207 | ) -> None: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 208 | pass |
| 209 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 210 | def visit_object_type( |
| 211 | self, |
| 212 | name: str, |
| 213 | info: Optional[QAPISourceInfo], |
| 214 | ifcond: QAPISchemaIfCond, |
| 215 | features: List[QAPISchemaFeature], |
| 216 | base: Optional[QAPISchemaObjectType], |
| 217 | members: List[QAPISchemaObjectTypeMember], |
Markus Armbruster | d1da8af | 2024-03-15 16:28:22 +0100 | [diff] [blame] | 218 | branches: Optional[QAPISchemaBranches], |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 219 | ) -> None: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 220 | pass |
| 221 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 222 | def visit_object_type_flat( |
| 223 | self, |
| 224 | name: str, |
| 225 | info: Optional[QAPISourceInfo], |
| 226 | ifcond: QAPISchemaIfCond, |
| 227 | features: List[QAPISchemaFeature], |
| 228 | members: List[QAPISchemaObjectTypeMember], |
Markus Armbruster | d1da8af | 2024-03-15 16:28:22 +0100 | [diff] [blame] | 229 | branches: Optional[QAPISchemaBranches], |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 230 | ) -> None: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 231 | pass |
| 232 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 233 | def visit_alternate_type( |
| 234 | self, |
| 235 | name: str, |
| 236 | info: Optional[QAPISourceInfo], |
| 237 | ifcond: QAPISchemaIfCond, |
| 238 | features: List[QAPISchemaFeature], |
Markus Armbruster | 41d0ad1 | 2024-03-16 07:43:36 +0100 | [diff] [blame] | 239 | alternatives: QAPISchemaAlternatives, |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 240 | ) -> None: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 241 | pass |
| 242 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 243 | def visit_command( |
| 244 | self, |
| 245 | name: str, |
| 246 | info: Optional[QAPISourceInfo], |
| 247 | ifcond: QAPISchemaIfCond, |
| 248 | features: List[QAPISchemaFeature], |
| 249 | arg_type: Optional[QAPISchemaObjectType], |
| 250 | ret_type: Optional[QAPISchemaType], |
| 251 | gen: bool, |
| 252 | success_response: bool, |
| 253 | boxed: bool, |
| 254 | allow_oob: bool, |
| 255 | allow_preconfig: bool, |
| 256 | coroutine: bool, |
| 257 | ) -> None: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 258 | pass |
| 259 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 260 | def visit_event( |
| 261 | self, |
| 262 | name: str, |
| 263 | info: Optional[QAPISourceInfo], |
| 264 | ifcond: QAPISchemaIfCond, |
| 265 | features: List[QAPISchemaFeature], |
| 266 | arg_type: Optional[QAPISchemaObjectType], |
| 267 | boxed: bool, |
| 268 | ) -> None: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 269 | pass |
| 270 | |
| 271 | |
Markus Armbruster | baa310f | 2020-03-04 16:59:29 +0100 | [diff] [blame] | 272 | class QAPISchemaModule: |
John Snow | 39b2d83 | 2021-02-01 14:37:41 -0500 | [diff] [blame] | 273 | |
| 274 | BUILTIN_MODULE_NAME = './builtin' |
| 275 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 276 | def __init__(self, name: str): |
Markus Armbruster | a9f1dd7 | 2019-11-20 19:25:49 +0100 | [diff] [blame] | 277 | self.name = name |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 278 | self._entity_list: List[QAPISchemaEntity] = [] |
Markus Armbruster | 3e7fb58 | 2019-11-20 19:25:50 +0100 | [diff] [blame] | 279 | |
John Snow | 98967c2 | 2021-02-01 14:37:36 -0500 | [diff] [blame] | 280 | @staticmethod |
John Snow | e2bbc4e | 2021-02-01 14:37:39 -0500 | [diff] [blame] | 281 | def is_system_module(name: str) -> bool: |
John Snow | 98967c2 | 2021-02-01 14:37:36 -0500 | [diff] [blame] | 282 | """ |
| 283 | System modules are internally defined modules. |
| 284 | |
| 285 | Their names start with the "./" prefix. |
| 286 | """ |
John Snow | e2bbc4e | 2021-02-01 14:37:39 -0500 | [diff] [blame] | 287 | return name.startswith('./') |
John Snow | 98967c2 | 2021-02-01 14:37:36 -0500 | [diff] [blame] | 288 | |
| 289 | @classmethod |
John Snow | e2bbc4e | 2021-02-01 14:37:39 -0500 | [diff] [blame] | 290 | def is_user_module(cls, name: str) -> bool: |
John Snow | 98967c2 | 2021-02-01 14:37:36 -0500 | [diff] [blame] | 291 | """ |
| 292 | User modules are those defined by the user in qapi JSON files. |
| 293 | |
| 294 | They do not start with the "./" prefix. |
| 295 | """ |
| 296 | return not cls.is_system_module(name) |
| 297 | |
John Snow | 39b2d83 | 2021-02-01 14:37:41 -0500 | [diff] [blame] | 298 | @classmethod |
| 299 | def is_builtin_module(cls, name: str) -> bool: |
John Snow | 98967c2 | 2021-02-01 14:37:36 -0500 | [diff] [blame] | 300 | """ |
| 301 | The built-in module is a single System module for the built-in types. |
| 302 | |
John Snow | e2bbc4e | 2021-02-01 14:37:39 -0500 | [diff] [blame] | 303 | It is always "./builtin". |
John Snow | 98967c2 | 2021-02-01 14:37:36 -0500 | [diff] [blame] | 304 | """ |
John Snow | 39b2d83 | 2021-02-01 14:37:41 -0500 | [diff] [blame] | 305 | return name == cls.BUILTIN_MODULE_NAME |
John Snow | 98967c2 | 2021-02-01 14:37:36 -0500 | [diff] [blame] | 306 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 307 | def add_entity(self, ent: QAPISchemaEntity) -> None: |
Markus Armbruster | 3e7fb58 | 2019-11-20 19:25:50 +0100 | [diff] [blame] | 308 | self._entity_list.append(ent) |
| 309 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 310 | def visit(self, visitor: QAPISchemaVisitor) -> None: |
Markus Armbruster | 3e7fb58 | 2019-11-20 19:25:50 +0100 | [diff] [blame] | 311 | visitor.visit_module(self.name) |
| 312 | for entity in self._entity_list: |
| 313 | if visitor.visit_needed(entity): |
| 314 | entity.visit(visitor) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 315 | |
Markus Armbruster | a9f1dd7 | 2019-11-20 19:25:49 +0100 | [diff] [blame] | 316 | |
| 317 | class QAPISchemaInclude(QAPISchemaEntity): |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 318 | def __init__(self, sub_module: QAPISchemaModule, info: QAPISourceInfo): |
John Snow | 2418d1c | 2024-03-15 16:22:41 +0100 | [diff] [blame] | 319 | super().__init__(info) |
Markus Armbruster | a9f1dd7 | 2019-11-20 19:25:49 +0100 | [diff] [blame] | 320 | self._sub_module = sub_module |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 321 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 322 | def visit(self, visitor: QAPISchemaVisitor) -> None: |
Markus Armbruster | 2cae67b | 2020-03-04 16:59:31 +0100 | [diff] [blame] | 323 | super().visit(visitor) |
Markus Armbruster | a9f1dd7 | 2019-11-20 19:25:49 +0100 | [diff] [blame] | 324 | visitor.visit_include(self._sub_module.name, self.info) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 325 | |
| 326 | |
John Snow | d150be3 | 2024-03-15 16:22:44 +0100 | [diff] [blame] | 327 | class QAPISchemaType(QAPISchemaDefinition, ABC): |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 328 | # Return the C type for common use. |
| 329 | # For the types we commonly box, this is a pointer type. |
John Snow | d150be3 | 2024-03-15 16:22:44 +0100 | [diff] [blame] | 330 | @abstractmethod |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 331 | def c_type(self) -> str: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 332 | pass |
| 333 | |
| 334 | # Return the C type to be used in a parameter list. |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 335 | def c_param_type(self) -> str: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 336 | return self.c_type() |
| 337 | |
| 338 | # Return the C type to be used where we suppress boxing. |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 339 | def c_unboxed_type(self) -> str: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 340 | return self.c_type() |
| 341 | |
John Snow | d150be3 | 2024-03-15 16:22:44 +0100 | [diff] [blame] | 342 | @abstractmethod |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 343 | def json_type(self) -> str: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 344 | pass |
| 345 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 346 | def alternate_qtype(self) -> Optional[str]: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 347 | json2qtype = { |
| 348 | 'null': 'QTYPE_QNULL', |
| 349 | 'string': 'QTYPE_QSTRING', |
| 350 | 'number': 'QTYPE_QNUM', |
| 351 | 'int': 'QTYPE_QNUM', |
| 352 | 'boolean': 'QTYPE_QBOOL', |
Paolo Bonzini | a580694 | 2022-03-21 17:42:41 +0100 | [diff] [blame] | 353 | 'array': 'QTYPE_QLIST', |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 354 | 'object': 'QTYPE_QDICT' |
| 355 | } |
| 356 | return json2qtype.get(self.json_type()) |
| 357 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 358 | def doc_type(self) -> Optional[str]: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 359 | if self.is_implicit(): |
| 360 | return None |
| 361 | return self.name |
| 362 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 363 | def need_has_if_optional(self) -> bool: |
Markus Armbruster | 44ea9d9 | 2022-11-04 17:06:46 +0100 | [diff] [blame] | 364 | # When FOO is a pointer, has_FOO == !!FOO, i.e. has_FOO is redundant. |
| 365 | # Except for arrays; see QAPISchemaArrayType.need_has_if_optional(). |
| 366 | return not self.c_type().endswith(POINTER_SUFFIX) |
| 367 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 368 | def check(self, schema: QAPISchema) -> None: |
Markus Armbruster | ecee568 | 2023-03-16 08:13:13 +0100 | [diff] [blame] | 369 | super().check(schema) |
Markus Armbruster | 57df0df | 2021-10-28 12:25:20 +0200 | [diff] [blame] | 370 | for feat in self.features: |
| 371 | if feat.is_special(): |
| 372 | raise QAPISemError( |
| 373 | self.info, |
| 374 | f"feature '{feat.name}' is not supported for types") |
Markus Armbruster | f965e8f | 2020-03-17 12:54:50 +0100 | [diff] [blame] | 375 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 376 | def describe(self) -> str: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 377 | return "%s type '%s'" % (self.meta, self.name) |
| 378 | |
| 379 | |
| 380 | class QAPISchemaBuiltinType(QAPISchemaType): |
| 381 | meta = 'built-in' |
| 382 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 383 | def __init__(self, name: str, json_type: str, c_type: str): |
Markus Armbruster | 2cae67b | 2020-03-04 16:59:31 +0100 | [diff] [blame] | 384 | super().__init__(name, None, None) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 385 | assert json_type in ('string', 'number', 'int', 'boolean', 'null', |
| 386 | 'value') |
| 387 | self._json_type_name = json_type |
| 388 | self._c_type_name = c_type |
| 389 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 390 | def c_name(self) -> str: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 391 | return self.name |
| 392 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 393 | def c_type(self) -> str: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 394 | return self._c_type_name |
| 395 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 396 | def c_param_type(self) -> str: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 397 | if self.name == 'str': |
| 398 | return 'const ' + self._c_type_name |
| 399 | return self._c_type_name |
| 400 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 401 | def json_type(self) -> str: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 402 | return self._json_type_name |
| 403 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 404 | def doc_type(self) -> str: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 405 | return self.json_type() |
| 406 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 407 | def visit(self, visitor: QAPISchemaVisitor) -> None: |
Markus Armbruster | 2cae67b | 2020-03-04 16:59:31 +0100 | [diff] [blame] | 408 | super().visit(visitor) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 409 | visitor.visit_builtin_type(self.name, self.info, self.json_type()) |
| 410 | |
| 411 | |
| 412 | class QAPISchemaEnumType(QAPISchemaType): |
| 413 | meta = 'enum' |
| 414 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 415 | def __init__( |
| 416 | self, |
| 417 | name: str, |
| 418 | info: Optional[QAPISourceInfo], |
| 419 | doc: Optional[QAPIDoc], |
| 420 | ifcond: Optional[QAPISchemaIfCond], |
| 421 | features: Optional[List[QAPISchemaFeature]], |
| 422 | members: List[QAPISchemaEnumMember], |
| 423 | prefix: Optional[str], |
| 424 | ): |
Markus Armbruster | 013b4ef | 2020-03-17 12:54:37 +0100 | [diff] [blame] | 425 | super().__init__(name, info, doc, ifcond, features) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 426 | for m in members: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 427 | m.set_defined_in(name) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 428 | self.members = members |
| 429 | self.prefix = prefix |
| 430 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 431 | def check(self, schema: QAPISchema) -> None: |
Markus Armbruster | 2cae67b | 2020-03-04 16:59:31 +0100 | [diff] [blame] | 432 | super().check(schema) |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 433 | seen: Dict[str, QAPISchemaMember] = {} |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 434 | for m in self.members: |
| 435 | m.check_clash(self.info, seen) |
Markus Armbruster | ee1e6a1 | 2019-10-24 13:02:26 +0200 | [diff] [blame] | 436 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 437 | def connect_doc(self, doc: Optional[QAPIDoc] = None) -> None: |
Markus Armbruster | e4405b3 | 2020-03-17 12:54:36 +0100 | [diff] [blame] | 438 | super().connect_doc(doc) |
Markus Armbruster | 7faefad | 2019-10-24 13:02:28 +0200 | [diff] [blame] | 439 | doc = doc or self.doc |
Markus Armbruster | 645178c | 2020-03-17 12:54:44 +0100 | [diff] [blame] | 440 | for m in self.members: |
| 441 | m.connect_doc(doc) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 442 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 443 | def is_implicit(self) -> bool: |
Markus Armbruster | 4e99f4b | 2021-09-17 16:31:32 +0200 | [diff] [blame] | 444 | # See QAPISchema._def_predefineds() |
| 445 | return self.name == 'QType' |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 446 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 447 | def c_type(self) -> str: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 448 | return c_name(self.name) |
| 449 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 450 | def member_names(self) -> List[str]: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 451 | return [m.name for m in self.members] |
| 452 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 453 | def json_type(self) -> str: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 454 | return 'string' |
| 455 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 456 | def visit(self, visitor: QAPISchemaVisitor) -> None: |
Markus Armbruster | 2cae67b | 2020-03-04 16:59:31 +0100 | [diff] [blame] | 457 | super().visit(visitor) |
Markus Armbruster | 013b4ef | 2020-03-17 12:54:37 +0100 | [diff] [blame] | 458 | visitor.visit_enum_type( |
| 459 | self.name, self.info, self.ifcond, self.features, |
| 460 | self.members, self.prefix) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 461 | |
| 462 | |
| 463 | class QAPISchemaArrayType(QAPISchemaType): |
| 464 | meta = 'array' |
| 465 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 466 | def __init__( |
| 467 | self, name: str, info: Optional[QAPISourceInfo], element_type: str |
| 468 | ): |
Markus Armbruster | 013b4ef | 2020-03-17 12:54:37 +0100 | [diff] [blame] | 469 | super().__init__(name, info, None) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 470 | self._element_type_name = element_type |
John Snow | 578cd93 | 2024-03-15 16:22:43 +0100 | [diff] [blame] | 471 | self.element_type: QAPISchemaType |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 472 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 473 | def need_has_if_optional(self) -> bool: |
Markus Armbruster | 44ea9d9 | 2022-11-04 17:06:46 +0100 | [diff] [blame] | 474 | # When FOO is an array, we still need has_FOO to distinguish |
| 475 | # absent (!has_FOO) from present and empty (has_FOO && !FOO). |
| 476 | return True |
| 477 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 478 | def check(self, schema: QAPISchema) -> None: |
Markus Armbruster | 2cae67b | 2020-03-04 16:59:31 +0100 | [diff] [blame] | 479 | super().check(schema) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 480 | self.element_type = schema.resolve_type( |
| 481 | self._element_type_name, self.info, |
John Snow | 8c91329 | 2024-03-15 16:22:49 +0100 | [diff] [blame] | 482 | self.info.defn_meta if self.info else None) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 483 | assert not isinstance(self.element_type, QAPISchemaArrayType) |
| 484 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 485 | def set_module(self, schema: QAPISchema) -> None: |
Markus Armbruster | a9f1dd7 | 2019-11-20 19:25:49 +0100 | [diff] [blame] | 486 | self._set_module(schema, self.element_type.info) |
| 487 | |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 488 | @property |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 489 | def ifcond(self) -> QAPISchemaIfCond: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 490 | assert self._checked |
| 491 | return self.element_type.ifcond |
| 492 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 493 | def is_implicit(self) -> bool: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 494 | return True |
| 495 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 496 | def c_type(self) -> str: |
John Snow | a7aa64a | 2020-10-09 12:15:34 -0400 | [diff] [blame] | 497 | return c_name(self.name) + POINTER_SUFFIX |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 498 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 499 | def json_type(self) -> str: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 500 | return 'array' |
| 501 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 502 | def doc_type(self) -> Optional[str]: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 503 | elt_doc_type = self.element_type.doc_type() |
| 504 | if not elt_doc_type: |
| 505 | return None |
| 506 | return 'array of ' + elt_doc_type |
| 507 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 508 | def visit(self, visitor: QAPISchemaVisitor) -> None: |
Markus Armbruster | 2cae67b | 2020-03-04 16:59:31 +0100 | [diff] [blame] | 509 | super().visit(visitor) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 510 | visitor.visit_array_type(self.name, self.info, self.ifcond, |
| 511 | self.element_type) |
| 512 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 513 | def describe(self) -> str: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 514 | return "%s type ['%s']" % (self.meta, self._element_type_name) |
| 515 | |
| 516 | |
| 517 | class QAPISchemaObjectType(QAPISchemaType): |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 518 | def __init__( |
| 519 | self, |
| 520 | name: str, |
| 521 | info: Optional[QAPISourceInfo], |
| 522 | doc: Optional[QAPIDoc], |
| 523 | ifcond: Optional[QAPISchemaIfCond], |
| 524 | features: Optional[List[QAPISchemaFeature]], |
| 525 | base: Optional[str], |
| 526 | local_members: List[QAPISchemaObjectTypeMember], |
Markus Armbruster | 3ff2a5a | 2024-03-15 16:33:23 +0100 | [diff] [blame] | 527 | branches: Optional[QAPISchemaBranches], |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 528 | ): |
Markus Armbruster | 3ff2a5a | 2024-03-15 16:33:23 +0100 | [diff] [blame] | 529 | # struct has local_members, optional base, and no branches |
| 530 | # union has base, branches, and no local_members |
Markus Armbruster | 2cae67b | 2020-03-04 16:59:31 +0100 | [diff] [blame] | 531 | super().__init__(name, info, doc, ifcond, features) |
Markus Armbruster | 3ff2a5a | 2024-03-15 16:33:23 +0100 | [diff] [blame] | 532 | self.meta = 'union' if branches else 'struct' |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 533 | for m in local_members: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 534 | m.set_defined_in(name) |
Markus Armbruster | 3ff2a5a | 2024-03-15 16:33:23 +0100 | [diff] [blame] | 535 | if branches is not None: |
| 536 | branches.set_defined_in(name) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 537 | self._base_name = base |
| 538 | self.base = None |
| 539 | self.local_members = local_members |
Markus Armbruster | 3ff2a5a | 2024-03-15 16:33:23 +0100 | [diff] [blame] | 540 | self.branches = branches |
John Snow | 9beda22 | 2024-03-15 16:22:52 +0100 | [diff] [blame] | 541 | self.members: List[QAPISchemaObjectTypeMember] |
John Snow | 875f624 | 2024-03-15 16:22:51 +0100 | [diff] [blame] | 542 | self._check_complete = False |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 543 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 544 | def check(self, schema: QAPISchema) -> None: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 545 | # This calls another type T's .check() exactly when the C |
| 546 | # struct emitted by gen_object() contains that T's C struct |
| 547 | # (pointers don't count). |
John Snow | 875f624 | 2024-03-15 16:22:51 +0100 | [diff] [blame] | 548 | if self._check_complete: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 549 | # A previous .check() completed: nothing to do |
| 550 | return |
| 551 | if self._checked: |
| 552 | # Recursed: C struct contains itself |
| 553 | raise QAPISemError(self.info, |
| 554 | "object %s contains itself" % self.name) |
| 555 | |
Markus Armbruster | 2cae67b | 2020-03-04 16:59:31 +0100 | [diff] [blame] | 556 | super().check(schema) |
John Snow | 875f624 | 2024-03-15 16:22:51 +0100 | [diff] [blame] | 557 | assert self._checked and not self._check_complete |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 558 | |
| 559 | seen = OrderedDict() |
| 560 | if self._base_name: |
| 561 | self.base = schema.resolve_type(self._base_name, self.info, |
| 562 | "'base'") |
| 563 | if (not isinstance(self.base, QAPISchemaObjectType) |
Markus Armbruster | 3ff2a5a | 2024-03-15 16:33:23 +0100 | [diff] [blame] | 564 | or self.base.branches): |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 565 | raise QAPISemError( |
| 566 | self.info, |
| 567 | "'base' requires a struct type, %s isn't" |
| 568 | % self.base.describe()) |
| 569 | self.base.check(schema) |
| 570 | self.base.check_clash(self.info, seen) |
| 571 | for m in self.local_members: |
| 572 | m.check(schema) |
| 573 | m.check_clash(self.info, seen) |
John Snow | 9beda22 | 2024-03-15 16:22:52 +0100 | [diff] [blame] | 574 | |
| 575 | # self.check_clash() works in terms of the supertype, but |
| 576 | # self.members is declared List[QAPISchemaObjectTypeMember]. |
| 577 | # Cast down to the subtype. |
| 578 | members = cast(List[QAPISchemaObjectTypeMember], list(seen.values())) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 579 | |
Markus Armbruster | 3ff2a5a | 2024-03-15 16:33:23 +0100 | [diff] [blame] | 580 | if self.branches: |
| 581 | self.branches.check(schema, seen) |
| 582 | self.branches.check_clash(self.info, seen) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 583 | |
John Snow | 875f624 | 2024-03-15 16:22:51 +0100 | [diff] [blame] | 584 | self.members = members |
| 585 | self._check_complete = True # mark completed |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 586 | |
| 587 | # Check that the members of this type do not cause duplicate JSON members, |
| 588 | # and update seen to track the members seen so far. Report any errors |
| 589 | # on behalf of info, which is not necessarily self.info |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 590 | def check_clash( |
| 591 | self, |
| 592 | info: Optional[QAPISourceInfo], |
| 593 | seen: Dict[str, QAPISchemaMember], |
| 594 | ) -> None: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 595 | assert self._checked |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 596 | for m in self.members: |
| 597 | m.check_clash(info, seen) |
Markus Armbruster | 3ff2a5a | 2024-03-15 16:33:23 +0100 | [diff] [blame] | 598 | if self.branches: |
| 599 | self.branches.check_clash(info, seen) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 600 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 601 | def connect_doc(self, doc: Optional[QAPIDoc] = None) -> None: |
Markus Armbruster | e4405b3 | 2020-03-17 12:54:36 +0100 | [diff] [blame] | 602 | super().connect_doc(doc) |
Markus Armbruster | 7faefad | 2019-10-24 13:02:28 +0200 | [diff] [blame] | 603 | doc = doc or self.doc |
Markus Armbruster | 645178c | 2020-03-17 12:54:44 +0100 | [diff] [blame] | 604 | if self.base and self.base.is_implicit(): |
| 605 | self.base.connect_doc(doc) |
| 606 | for m in self.local_members: |
| 607 | m.connect_doc(doc) |
Markus Armbruster | ee1e6a1 | 2019-10-24 13:02:26 +0200 | [diff] [blame] | 608 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 609 | def is_implicit(self) -> bool: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 610 | # See QAPISchema._make_implicit_object_type(), as well as |
| 611 | # _def_predefineds() |
| 612 | return self.name.startswith('q_') |
| 613 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 614 | def is_empty(self) -> bool: |
Markus Armbruster | 3ff2a5a | 2024-03-15 16:33:23 +0100 | [diff] [blame] | 615 | return not self.members and not self.branches |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 616 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 617 | def has_conditional_members(self) -> bool: |
Markus Armbruster | de3b3f5 | 2023-03-16 08:13:25 +0100 | [diff] [blame] | 618 | return any(m.ifcond.is_present() for m in self.members) |
| 619 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 620 | def c_name(self) -> str: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 621 | assert self.name != 'q_empty' |
Markus Armbruster | 2cae67b | 2020-03-04 16:59:31 +0100 | [diff] [blame] | 622 | return super().c_name() |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 623 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 624 | def c_type(self) -> str: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 625 | assert not self.is_implicit() |
John Snow | a7aa64a | 2020-10-09 12:15:34 -0400 | [diff] [blame] | 626 | return c_name(self.name) + POINTER_SUFFIX |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 627 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 628 | def c_unboxed_type(self) -> str: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 629 | return c_name(self.name) |
| 630 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 631 | def json_type(self) -> str: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 632 | return 'object' |
| 633 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 634 | def visit(self, visitor: QAPISchemaVisitor) -> None: |
Markus Armbruster | 2cae67b | 2020-03-04 16:59:31 +0100 | [diff] [blame] | 635 | super().visit(visitor) |
Markus Armbruster | 7b3bc9e | 2020-03-17 12:54:38 +0100 | [diff] [blame] | 636 | visitor.visit_object_type( |
| 637 | self.name, self.info, self.ifcond, self.features, |
Markus Armbruster | 3ff2a5a | 2024-03-15 16:33:23 +0100 | [diff] [blame] | 638 | self.base, self.local_members, self.branches) |
Markus Armbruster | 7b3bc9e | 2020-03-17 12:54:38 +0100 | [diff] [blame] | 639 | visitor.visit_object_type_flat( |
| 640 | self.name, self.info, self.ifcond, self.features, |
Markus Armbruster | 3ff2a5a | 2024-03-15 16:33:23 +0100 | [diff] [blame] | 641 | self.members, self.branches) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 642 | |
| 643 | |
Markus Armbruster | 226b5be | 2020-03-17 12:54:42 +0100 | [diff] [blame] | 644 | class QAPISchemaAlternateType(QAPISchemaType): |
| 645 | meta = 'alternate' |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 646 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 647 | def __init__( |
| 648 | self, |
| 649 | name: str, |
| 650 | info: QAPISourceInfo, |
| 651 | doc: Optional[QAPIDoc], |
| 652 | ifcond: Optional[QAPISchemaIfCond], |
| 653 | features: List[QAPISchemaFeature], |
Markus Armbruster | e0a28f3 | 2024-03-15 16:36:02 +0100 | [diff] [blame] | 654 | alternatives: QAPISchemaAlternatives, |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 655 | ): |
Markus Armbruster | 226b5be | 2020-03-17 12:54:42 +0100 | [diff] [blame] | 656 | super().__init__(name, info, doc, ifcond, features) |
Markus Armbruster | e0a28f3 | 2024-03-15 16:36:02 +0100 | [diff] [blame] | 657 | assert alternatives.tag_member |
| 658 | alternatives.set_defined_in(name) |
| 659 | alternatives.tag_member.set_defined_in(self.name) |
| 660 | self.alternatives = alternatives |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 661 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 662 | def check(self, schema: QAPISchema) -> None: |
Markus Armbruster | 226b5be | 2020-03-17 12:54:42 +0100 | [diff] [blame] | 663 | super().check(schema) |
Markus Armbruster | e0a28f3 | 2024-03-15 16:36:02 +0100 | [diff] [blame] | 664 | self.alternatives.tag_member.check(schema) |
| 665 | # Not calling self.alternatives.check_clash(), because there's |
| 666 | # nothing to clash with |
| 667 | self.alternatives.check(schema, {}) |
Markus Armbruster | 226b5be | 2020-03-17 12:54:42 +0100 | [diff] [blame] | 668 | # Alternate branch names have no relation to the tag enum values; |
| 669 | # so we have to check for potential name collisions ourselves. |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 670 | seen: Dict[str, QAPISchemaMember] = {} |
| 671 | types_seen: Dict[str, str] = {} |
Markus Armbruster | e0a28f3 | 2024-03-15 16:36:02 +0100 | [diff] [blame] | 672 | for v in self.alternatives.variants: |
Markus Armbruster | 226b5be | 2020-03-17 12:54:42 +0100 | [diff] [blame] | 673 | v.check_clash(self.info, seen) |
| 674 | qtype = v.type.alternate_qtype() |
| 675 | if not qtype: |
| 676 | raise QAPISemError( |
| 677 | self.info, |
| 678 | "%s cannot use %s" |
| 679 | % (v.describe(self.info), v.type.describe())) |
| 680 | conflicting = set([qtype]) |
| 681 | if qtype == 'QTYPE_QSTRING': |
| 682 | if isinstance(v.type, QAPISchemaEnumType): |
| 683 | for m in v.type.members: |
| 684 | if m.name in ['on', 'off']: |
| 685 | conflicting.add('QTYPE_QBOOL') |
| 686 | if re.match(r'[-+0-9.]', m.name): |
| 687 | # lazy, could be tightened |
| 688 | conflicting.add('QTYPE_QNUM') |
| 689 | else: |
| 690 | conflicting.add('QTYPE_QNUM') |
| 691 | conflicting.add('QTYPE_QBOOL') |
| 692 | for qt in conflicting: |
| 693 | if qt in types_seen: |
| 694 | raise QAPISemError( |
| 695 | self.info, |
| 696 | "%s can't be distinguished from '%s'" |
| 697 | % (v.describe(self.info), types_seen[qt])) |
| 698 | types_seen[qt] = v.name |
| 699 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 700 | def connect_doc(self, doc: Optional[QAPIDoc] = None) -> None: |
Markus Armbruster | 226b5be | 2020-03-17 12:54:42 +0100 | [diff] [blame] | 701 | super().connect_doc(doc) |
| 702 | doc = doc or self.doc |
Markus Armbruster | e0a28f3 | 2024-03-15 16:36:02 +0100 | [diff] [blame] | 703 | for v in self.alternatives.variants: |
Markus Armbruster | 645178c | 2020-03-17 12:54:44 +0100 | [diff] [blame] | 704 | v.connect_doc(doc) |
Markus Armbruster | 226b5be | 2020-03-17 12:54:42 +0100 | [diff] [blame] | 705 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 706 | def c_type(self) -> str: |
John Snow | a7aa64a | 2020-10-09 12:15:34 -0400 | [diff] [blame] | 707 | return c_name(self.name) + POINTER_SUFFIX |
Markus Armbruster | 226b5be | 2020-03-17 12:54:42 +0100 | [diff] [blame] | 708 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 709 | def json_type(self) -> str: |
Markus Armbruster | 226b5be | 2020-03-17 12:54:42 +0100 | [diff] [blame] | 710 | return 'value' |
| 711 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 712 | def visit(self, visitor: QAPISchemaVisitor) -> None: |
Markus Armbruster | 226b5be | 2020-03-17 12:54:42 +0100 | [diff] [blame] | 713 | super().visit(visitor) |
| 714 | visitor.visit_alternate_type( |
Markus Armbruster | e0a28f3 | 2024-03-15 16:36:02 +0100 | [diff] [blame] | 715 | self.name, self.info, self.ifcond, self.features, |
| 716 | self.alternatives) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 717 | |
| 718 | |
Markus Armbruster | 5858fd1 | 2020-03-17 12:54:43 +0100 | [diff] [blame] | 719 | class QAPISchemaVariants: |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 720 | def __init__( |
| 721 | self, |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 722 | info: QAPISourceInfo, |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 723 | variants: List[QAPISchemaVariant], |
| 724 | ): |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 725 | self.info = info |
Markus Armbruster | 285a8f2 | 2024-03-16 08:46:12 +0100 | [diff] [blame] | 726 | self.tag_member: QAPISchemaObjectTypeMember |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 727 | self.variants = variants |
| 728 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 729 | def set_defined_in(self, name: str) -> None: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 730 | for v in self.variants: |
| 731 | v.set_defined_in(name) |
| 732 | |
John Snow | f64e753 | 2024-06-26 18:21:08 -0400 | [diff] [blame] | 733 | # pylint: disable=unused-argument |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 734 | def check( |
Markus Armbruster | 8152bc7 | 2024-03-15 20:57:56 +0100 | [diff] [blame] | 735 | self, schema: QAPISchema, seen: Dict[str, QAPISchemaMember] |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 736 | ) -> None: |
Markus Armbruster | 8152bc7 | 2024-03-15 20:57:56 +0100 | [diff] [blame] | 737 | for v in self.variants: |
| 738 | v.check(schema) |
John Snow | 583f4d6 | 2024-03-15 16:22:53 +0100 | [diff] [blame] | 739 | |
Markus Armbruster | 8152bc7 | 2024-03-15 20:57:56 +0100 | [diff] [blame] | 740 | |
| 741 | class QAPISchemaBranches(QAPISchemaVariants): |
| 742 | def __init__(self, |
| 743 | info: QAPISourceInfo, |
| 744 | variants: List[QAPISchemaVariant], |
| 745 | tag_name: str): |
| 746 | super().__init__(info, variants) |
| 747 | self._tag_name = tag_name |
| 748 | |
| 749 | def check( |
| 750 | self, schema: QAPISchema, seen: Dict[str, QAPISchemaMember] |
| 751 | ) -> None: |
| 752 | # We need to narrow the member type: |
Markus Armbruster | 285a8f2 | 2024-03-16 08:46:12 +0100 | [diff] [blame] | 753 | tag_member = seen.get(c_name(self._tag_name)) |
| 754 | assert (tag_member is None |
| 755 | or isinstance(tag_member, QAPISchemaObjectTypeMember)) |
Markus Armbruster | 8152bc7 | 2024-03-15 20:57:56 +0100 | [diff] [blame] | 756 | |
| 757 | base = "'base'" |
| 758 | # Pointing to the base type when not implicit would be |
| 759 | # nice, but we don't know it here |
Markus Armbruster | 285a8f2 | 2024-03-16 08:46:12 +0100 | [diff] [blame] | 760 | if not tag_member or self._tag_name != tag_member.name: |
Markus Armbruster | 8152bc7 | 2024-03-15 20:57:56 +0100 | [diff] [blame] | 761 | raise QAPISemError( |
| 762 | self.info, |
| 763 | "discriminator '%s' is not a member of %s" |
| 764 | % (self._tag_name, base)) |
Markus Armbruster | 285a8f2 | 2024-03-16 08:46:12 +0100 | [diff] [blame] | 765 | self.tag_member = tag_member |
Markus Armbruster | 8152bc7 | 2024-03-15 20:57:56 +0100 | [diff] [blame] | 766 | # Here we do: |
Markus Armbruster | 285a8f2 | 2024-03-16 08:46:12 +0100 | [diff] [blame] | 767 | assert tag_member.defined_in |
| 768 | base_type = schema.lookup_type(tag_member.defined_in) |
Markus Armbruster | 8152bc7 | 2024-03-15 20:57:56 +0100 | [diff] [blame] | 769 | assert base_type |
| 770 | if not base_type.is_implicit(): |
Markus Armbruster | 285a8f2 | 2024-03-16 08:46:12 +0100 | [diff] [blame] | 771 | base = "base type '%s'" % tag_member.defined_in |
| 772 | if not isinstance(tag_member.type, QAPISchemaEnumType): |
Markus Armbruster | 8152bc7 | 2024-03-15 20:57:56 +0100 | [diff] [blame] | 773 | raise QAPISemError( |
| 774 | self.info, |
| 775 | "discriminator member '%s' of %s must be of enum type" |
| 776 | % (self._tag_name, base)) |
Markus Armbruster | 285a8f2 | 2024-03-16 08:46:12 +0100 | [diff] [blame] | 777 | if tag_member.optional: |
Markus Armbruster | 8152bc7 | 2024-03-15 20:57:56 +0100 | [diff] [blame] | 778 | raise QAPISemError( |
| 779 | self.info, |
| 780 | "discriminator member '%s' of %s must not be optional" |
| 781 | % (self._tag_name, base)) |
Markus Armbruster | 285a8f2 | 2024-03-16 08:46:12 +0100 | [diff] [blame] | 782 | if tag_member.ifcond.is_present(): |
Markus Armbruster | 8152bc7 | 2024-03-15 20:57:56 +0100 | [diff] [blame] | 783 | raise QAPISemError( |
| 784 | self.info, |
| 785 | "discriminator member '%s' of %s must not be conditional" |
| 786 | % (self._tag_name, base)) |
| 787 | # branches that are not explicitly covered get an empty type |
Markus Armbruster | 285a8f2 | 2024-03-16 08:46:12 +0100 | [diff] [blame] | 788 | assert tag_member.defined_in |
Markus Armbruster | 8152bc7 | 2024-03-15 20:57:56 +0100 | [diff] [blame] | 789 | cases = {v.name for v in self.variants} |
Markus Armbruster | 285a8f2 | 2024-03-16 08:46:12 +0100 | [diff] [blame] | 790 | for m in tag_member.type.members: |
Markus Armbruster | 8152bc7 | 2024-03-15 20:57:56 +0100 | [diff] [blame] | 791 | if m.name not in cases: |
| 792 | v = QAPISchemaVariant(m.name, self.info, |
| 793 | 'q_empty', m.ifcond) |
Markus Armbruster | 285a8f2 | 2024-03-16 08:46:12 +0100 | [diff] [blame] | 794 | v.set_defined_in(tag_member.defined_in) |
Markus Armbruster | 8152bc7 | 2024-03-15 20:57:56 +0100 | [diff] [blame] | 795 | self.variants.append(v) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 796 | if not self.variants: |
| 797 | raise QAPISemError(self.info, "union has no branches") |
| 798 | for v in self.variants: |
| 799 | v.check(schema) |
| 800 | # Union names must match enum values; alternate names are |
| 801 | # checked separately. Use 'seen' to tell the two apart. |
| 802 | if seen: |
Markus Armbruster | 285a8f2 | 2024-03-16 08:46:12 +0100 | [diff] [blame] | 803 | if v.name not in tag_member.type.member_names(): |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 804 | raise QAPISemError( |
| 805 | self.info, |
| 806 | "branch '%s' is not a value of %s" |
Markus Armbruster | 285a8f2 | 2024-03-16 08:46:12 +0100 | [diff] [blame] | 807 | % (v.name, tag_member.type.describe())) |
Daniel P. Berrangé | a17dbc4 | 2023-04-20 11:26:19 +0100 | [diff] [blame] | 808 | if not isinstance(v.type, QAPISchemaObjectType): |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 809 | raise QAPISemError( |
| 810 | self.info, |
| 811 | "%s cannot use %s" |
| 812 | % (v.describe(self.info), v.type.describe())) |
| 813 | v.type.check(schema) |
| 814 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 815 | def check_clash( |
| 816 | self, |
| 817 | info: Optional[QAPISourceInfo], |
| 818 | seen: Dict[str, QAPISchemaMember], |
| 819 | ) -> None: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 820 | for v in self.variants: |
| 821 | # Reset seen map for each variant, since qapi names from one |
John Snow | 7e09dd6 | 2024-03-15 16:22:54 +0100 | [diff] [blame] | 822 | # branch do not affect another branch. |
| 823 | # |
| 824 | # v.type's typing is enforced in check() above. |
| 825 | assert isinstance(v.type, QAPISchemaObjectType) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 826 | v.type.check_clash(info, dict(seen)) |
| 827 | |
| 828 | |
Markus Armbruster | 1d067e3 | 2024-03-15 20:32:41 +0100 | [diff] [blame] | 829 | class QAPISchemaAlternatives(QAPISchemaVariants): |
| 830 | def __init__(self, |
| 831 | info: QAPISourceInfo, |
| 832 | variants: List[QAPISchemaVariant], |
| 833 | tag_member: QAPISchemaObjectTypeMember): |
Markus Armbruster | 8152bc7 | 2024-03-15 20:57:56 +0100 | [diff] [blame] | 834 | super().__init__(info, variants) |
Markus Armbruster | 285a8f2 | 2024-03-16 08:46:12 +0100 | [diff] [blame] | 835 | self.tag_member = tag_member |
Markus Armbruster | 8152bc7 | 2024-03-15 20:57:56 +0100 | [diff] [blame] | 836 | |
| 837 | def check( |
| 838 | self, schema: QAPISchema, seen: Dict[str, QAPISchemaMember] |
| 839 | ) -> None: |
| 840 | super().check(schema, seen) |
| 841 | assert isinstance(self.tag_member.type, QAPISchemaEnumType) |
| 842 | assert not self.tag_member.optional |
| 843 | assert not self.tag_member.ifcond.is_present() |
Markus Armbruster | 1d067e3 | 2024-03-15 20:32:41 +0100 | [diff] [blame] | 844 | |
| 845 | |
Markus Armbruster | 226b5be | 2020-03-17 12:54:42 +0100 | [diff] [blame] | 846 | class QAPISchemaMember: |
| 847 | """ Represents object members, enum members and features """ |
| 848 | role = 'member' |
| 849 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 850 | def __init__( |
| 851 | self, |
| 852 | name: str, |
| 853 | info: Optional[QAPISourceInfo], |
| 854 | ifcond: Optional[QAPISchemaIfCond] = None, |
| 855 | ): |
Markus Armbruster | 226b5be | 2020-03-17 12:54:42 +0100 | [diff] [blame] | 856 | self.name = name |
| 857 | self.info = info |
Marc-André Lureau | f17539c | 2021-08-04 12:30:57 +0400 | [diff] [blame] | 858 | self.ifcond = ifcond or QAPISchemaIfCond() |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 859 | self.defined_in: Optional[str] = None |
Markus Armbruster | 226b5be | 2020-03-17 12:54:42 +0100 | [diff] [blame] | 860 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 861 | def set_defined_in(self, name: str) -> None: |
Markus Armbruster | 226b5be | 2020-03-17 12:54:42 +0100 | [diff] [blame] | 862 | assert not self.defined_in |
| 863 | self.defined_in = name |
| 864 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 865 | def check_clash( |
| 866 | self, |
| 867 | info: Optional[QAPISourceInfo], |
| 868 | seen: Dict[str, QAPISchemaMember], |
| 869 | ) -> None: |
Markus Armbruster | 226b5be | 2020-03-17 12:54:42 +0100 | [diff] [blame] | 870 | cname = c_name(self.name) |
| 871 | if cname in seen: |
| 872 | raise QAPISemError( |
| 873 | info, |
| 874 | "%s collides with %s" |
| 875 | % (self.describe(info), seen[cname].describe(info))) |
| 876 | seen[cname] = self |
| 877 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 878 | def connect_doc(self, doc: Optional[QAPIDoc]) -> None: |
Markus Armbruster | 645178c | 2020-03-17 12:54:44 +0100 | [diff] [blame] | 879 | if doc: |
| 880 | doc.connect_member(self) |
| 881 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 882 | def describe(self, info: Optional[QAPISourceInfo]) -> str: |
Markus Armbruster | 226b5be | 2020-03-17 12:54:42 +0100 | [diff] [blame] | 883 | role = self.role |
Markus Armbruster | 1e148b5 | 2023-04-25 15:10:28 +0200 | [diff] [blame] | 884 | meta = 'type' |
Markus Armbruster | 226b5be | 2020-03-17 12:54:42 +0100 | [diff] [blame] | 885 | defined_in = self.defined_in |
| 886 | assert defined_in |
| 887 | |
| 888 | if defined_in.startswith('q_obj_'): |
| 889 | # See QAPISchema._make_implicit_object_type() - reverse the |
| 890 | # mapping there to create a nice human-readable description |
| 891 | defined_in = defined_in[6:] |
| 892 | if defined_in.endswith('-arg'): |
| 893 | # Implicit type created for a command's dict 'data' |
| 894 | assert role == 'member' |
| 895 | role = 'parameter' |
Markus Armbruster | 1e148b5 | 2023-04-25 15:10:28 +0200 | [diff] [blame] | 896 | meta = 'command' |
| 897 | defined_in = defined_in[:-4] |
Markus Armbruster | 226b5be | 2020-03-17 12:54:42 +0100 | [diff] [blame] | 898 | elif defined_in.endswith('-base'): |
Markus Armbruster | 4e99f4b | 2021-09-17 16:31:32 +0200 | [diff] [blame] | 899 | # Implicit type created for a union's dict 'base' |
Markus Armbruster | 226b5be | 2020-03-17 12:54:42 +0100 | [diff] [blame] | 900 | role = 'base ' + role |
Markus Armbruster | 1e148b5 | 2023-04-25 15:10:28 +0200 | [diff] [blame] | 901 | defined_in = defined_in[:-5] |
Markus Armbruster | 226b5be | 2020-03-17 12:54:42 +0100 | [diff] [blame] | 902 | else: |
Markus Armbruster | 226b5be | 2020-03-17 12:54:42 +0100 | [diff] [blame] | 903 | assert False |
Markus Armbruster | 1e148b5 | 2023-04-25 15:10:28 +0200 | [diff] [blame] | 904 | |
John Snow | 8b9e7fd | 2024-03-15 16:22:50 +0100 | [diff] [blame] | 905 | assert info is not None |
Markus Armbruster | 1e148b5 | 2023-04-25 15:10:28 +0200 | [diff] [blame] | 906 | if defined_in != info.defn_name: |
| 907 | return "%s '%s' of %s '%s'" % (role, self.name, meta, defined_in) |
Markus Armbruster | 226b5be | 2020-03-17 12:54:42 +0100 | [diff] [blame] | 908 | return "%s '%s'" % (role, self.name) |
| 909 | |
| 910 | |
| 911 | class QAPISchemaEnumMember(QAPISchemaMember): |
| 912 | role = 'value' |
| 913 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 914 | def __init__( |
| 915 | self, |
| 916 | name: str, |
| 917 | info: Optional[QAPISourceInfo], |
| 918 | ifcond: Optional[QAPISchemaIfCond] = None, |
| 919 | features: Optional[List[QAPISchemaFeature]] = None, |
| 920 | ): |
Markus Armbruster | b6c1875 | 2021-10-25 06:24:02 +0200 | [diff] [blame] | 921 | super().__init__(name, info, ifcond) |
| 922 | for f in features or []: |
Markus Armbruster | b6c1875 | 2021-10-25 06:24:02 +0200 | [diff] [blame] | 923 | f.set_defined_in(name) |
| 924 | self.features = features or [] |
| 925 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 926 | def connect_doc(self, doc: Optional[QAPIDoc]) -> None: |
Markus Armbruster | b6c1875 | 2021-10-25 06:24:02 +0200 | [diff] [blame] | 927 | super().connect_doc(doc) |
| 928 | if doc: |
| 929 | for f in self.features: |
| 930 | doc.connect_feature(f) |
| 931 | |
Markus Armbruster | 226b5be | 2020-03-17 12:54:42 +0100 | [diff] [blame] | 932 | |
| 933 | class QAPISchemaFeature(QAPISchemaMember): |
| 934 | role = 'feature' |
| 935 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 936 | def is_special(self) -> bool: |
Markus Armbruster | 57df0df | 2021-10-28 12:25:20 +0200 | [diff] [blame] | 937 | return self.name in ('deprecated', 'unstable') |
Markus Armbruster | c67db1e | 2021-10-28 12:25:15 +0200 | [diff] [blame] | 938 | |
Markus Armbruster | 226b5be | 2020-03-17 12:54:42 +0100 | [diff] [blame] | 939 | |
| 940 | class QAPISchemaObjectTypeMember(QAPISchemaMember): |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 941 | def __init__( |
| 942 | self, |
| 943 | name: str, |
| 944 | info: QAPISourceInfo, |
| 945 | typ: str, |
| 946 | optional: bool, |
| 947 | ifcond: Optional[QAPISchemaIfCond] = None, |
| 948 | features: Optional[List[QAPISchemaFeature]] = None, |
| 949 | ): |
Markus Armbruster | 226b5be | 2020-03-17 12:54:42 +0100 | [diff] [blame] | 950 | super().__init__(name, info, ifcond) |
Markus Armbruster | 84ab008 | 2020-03-17 12:54:45 +0100 | [diff] [blame] | 951 | for f in features or []: |
Markus Armbruster | 84ab008 | 2020-03-17 12:54:45 +0100 | [diff] [blame] | 952 | f.set_defined_in(name) |
Markus Armbruster | 226b5be | 2020-03-17 12:54:42 +0100 | [diff] [blame] | 953 | self._type_name = typ |
John Snow | ec10396 | 2024-03-15 16:22:42 +0100 | [diff] [blame] | 954 | self.type: QAPISchemaType # set during check() |
Markus Armbruster | 226b5be | 2020-03-17 12:54:42 +0100 | [diff] [blame] | 955 | self.optional = optional |
Markus Armbruster | 84ab008 | 2020-03-17 12:54:45 +0100 | [diff] [blame] | 956 | self.features = features or [] |
Markus Armbruster | 226b5be | 2020-03-17 12:54:42 +0100 | [diff] [blame] | 957 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 958 | def need_has(self) -> bool: |
Markus Armbruster | 44ea9d9 | 2022-11-04 17:06:46 +0100 | [diff] [blame] | 959 | return self.optional and self.type.need_has_if_optional() |
| 960 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 961 | def check(self, schema: QAPISchema) -> None: |
Markus Armbruster | 226b5be | 2020-03-17 12:54:42 +0100 | [diff] [blame] | 962 | assert self.defined_in |
| 963 | self.type = schema.resolve_type(self._type_name, self.info, |
| 964 | self.describe) |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 965 | seen: Dict[str, QAPISchemaMember] = {} |
Markus Armbruster | 84ab008 | 2020-03-17 12:54:45 +0100 | [diff] [blame] | 966 | for f in self.features: |
| 967 | f.check_clash(self.info, seen) |
| 968 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 969 | def connect_doc(self, doc: Optional[QAPIDoc]) -> None: |
Markus Armbruster | 84ab008 | 2020-03-17 12:54:45 +0100 | [diff] [blame] | 970 | super().connect_doc(doc) |
| 971 | if doc: |
| 972 | for f in self.features: |
| 973 | doc.connect_feature(f) |
Markus Armbruster | 226b5be | 2020-03-17 12:54:42 +0100 | [diff] [blame] | 974 | |
| 975 | |
Markus Armbruster | 5858fd1 | 2020-03-17 12:54:43 +0100 | [diff] [blame] | 976 | class QAPISchemaVariant(QAPISchemaObjectTypeMember): |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 977 | role = 'branch' |
| 978 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 979 | def __init__( |
| 980 | self, |
| 981 | name: str, |
| 982 | info: QAPISourceInfo, |
| 983 | typ: str, |
| 984 | ifcond: QAPISchemaIfCond, |
| 985 | ): |
Markus Armbruster | 2cae67b | 2020-03-04 16:59:31 +0100 | [diff] [blame] | 986 | super().__init__(name, info, typ, False, ifcond) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 987 | |
| 988 | |
John Snow | 2418d1c | 2024-03-15 16:22:41 +0100 | [diff] [blame] | 989 | class QAPISchemaCommand(QAPISchemaDefinition): |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 990 | meta = 'command' |
| 991 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 992 | def __init__( |
| 993 | self, |
| 994 | name: str, |
| 995 | info: QAPISourceInfo, |
| 996 | doc: Optional[QAPIDoc], |
| 997 | ifcond: QAPISchemaIfCond, |
| 998 | features: List[QAPISchemaFeature], |
| 999 | arg_type: Optional[str], |
| 1000 | ret_type: Optional[str], |
| 1001 | gen: bool, |
| 1002 | success_response: bool, |
| 1003 | boxed: bool, |
| 1004 | allow_oob: bool, |
| 1005 | allow_preconfig: bool, |
| 1006 | coroutine: bool, |
| 1007 | ): |
Markus Armbruster | 2cae67b | 2020-03-04 16:59:31 +0100 | [diff] [blame] | 1008 | super().__init__(name, info, doc, ifcond, features) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1009 | self._arg_type_name = arg_type |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 1010 | self.arg_type: Optional[QAPISchemaObjectType] = None |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1011 | self._ret_type_name = ret_type |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 1012 | self.ret_type: Optional[QAPISchemaType] = None |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1013 | self.gen = gen |
| 1014 | self.success_response = success_response |
| 1015 | self.boxed = boxed |
| 1016 | self.allow_oob = allow_oob |
| 1017 | self.allow_preconfig = allow_preconfig |
Kevin Wolf | 04f2236 | 2020-10-05 17:58:49 +0200 | [diff] [blame] | 1018 | self.coroutine = coroutine |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1019 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 1020 | def check(self, schema: QAPISchema) -> None: |
John Snow | 8b9e7fd | 2024-03-15 16:22:50 +0100 | [diff] [blame] | 1021 | assert self.info is not None |
Markus Armbruster | 2cae67b | 2020-03-04 16:59:31 +0100 | [diff] [blame] | 1022 | super().check(schema) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1023 | if self._arg_type_name: |
John Snow | 9bda6c7 | 2024-03-15 16:22:45 +0100 | [diff] [blame] | 1024 | arg_type = schema.resolve_type( |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1025 | self._arg_type_name, self.info, "command's 'data'") |
John Snow | 9bda6c7 | 2024-03-15 16:22:45 +0100 | [diff] [blame] | 1026 | if not isinstance(arg_type, QAPISchemaObjectType): |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1027 | raise QAPISemError( |
| 1028 | self.info, |
| 1029 | "command's 'data' cannot take %s" |
John Snow | 9bda6c7 | 2024-03-15 16:22:45 +0100 | [diff] [blame] | 1030 | % arg_type.describe()) |
| 1031 | self.arg_type = arg_type |
Markus Armbruster | 3ff2a5a | 2024-03-15 16:33:23 +0100 | [diff] [blame] | 1032 | if self.arg_type.branches and not self.boxed: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1033 | raise QAPISemError( |
| 1034 | self.info, |
| 1035 | "command's 'data' can take %s only with 'boxed': true" |
| 1036 | % self.arg_type.describe()) |
Markus Armbruster | de3b3f5 | 2023-03-16 08:13:25 +0100 | [diff] [blame] | 1037 | self.arg_type.check(schema) |
| 1038 | if self.arg_type.has_conditional_members() and not self.boxed: |
| 1039 | raise QAPISemError( |
| 1040 | self.info, |
| 1041 | "conditional command arguments require 'boxed': true") |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1042 | if self._ret_type_name: |
| 1043 | self.ret_type = schema.resolve_type( |
| 1044 | self._ret_type_name, self.info, "command's 'returns'") |
Markus Armbruster | b86df37 | 2021-03-23 10:40:16 +0100 | [diff] [blame] | 1045 | if self.name not in self.info.pragma.command_returns_exceptions: |
Markus Armbruster | 7e9c170 | 2019-11-20 19:25:46 +0100 | [diff] [blame] | 1046 | typ = self.ret_type |
| 1047 | if isinstance(typ, QAPISchemaArrayType): |
John Snow | 9bda6c7 | 2024-03-15 16:22:45 +0100 | [diff] [blame] | 1048 | typ = typ.element_type |
Markus Armbruster | 7e9c170 | 2019-11-20 19:25:46 +0100 | [diff] [blame] | 1049 | if not isinstance(typ, QAPISchemaObjectType): |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1050 | raise QAPISemError( |
| 1051 | self.info, |
| 1052 | "command's 'returns' cannot take %s" |
| 1053 | % self.ret_type.describe()) |
| 1054 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 1055 | def connect_doc(self, doc: Optional[QAPIDoc] = None) -> None: |
Markus Armbruster | e4405b3 | 2020-03-17 12:54:36 +0100 | [diff] [blame] | 1056 | super().connect_doc(doc) |
Markus Armbruster | bf83f04 | 2019-10-24 13:02:29 +0200 | [diff] [blame] | 1057 | doc = doc or self.doc |
| 1058 | if doc: |
| 1059 | if self.arg_type and self.arg_type.is_implicit(): |
| 1060 | self.arg_type.connect_doc(doc) |
| 1061 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 1062 | def visit(self, visitor: QAPISchemaVisitor) -> None: |
Markus Armbruster | 2cae67b | 2020-03-04 16:59:31 +0100 | [diff] [blame] | 1063 | super().visit(visitor) |
Markus Armbruster | 7b3bc9e | 2020-03-17 12:54:38 +0100 | [diff] [blame] | 1064 | visitor.visit_command( |
| 1065 | self.name, self.info, self.ifcond, self.features, |
| 1066 | self.arg_type, self.ret_type, self.gen, self.success_response, |
Kevin Wolf | 04f2236 | 2020-10-05 17:58:49 +0200 | [diff] [blame] | 1067 | self.boxed, self.allow_oob, self.allow_preconfig, |
| 1068 | self.coroutine) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1069 | |
| 1070 | |
John Snow | 2418d1c | 2024-03-15 16:22:41 +0100 | [diff] [blame] | 1071 | class QAPISchemaEvent(QAPISchemaDefinition): |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1072 | meta = 'event' |
| 1073 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 1074 | def __init__( |
| 1075 | self, |
| 1076 | name: str, |
| 1077 | info: QAPISourceInfo, |
| 1078 | doc: Optional[QAPIDoc], |
| 1079 | ifcond: QAPISchemaIfCond, |
| 1080 | features: List[QAPISchemaFeature], |
| 1081 | arg_type: Optional[str], |
| 1082 | boxed: bool, |
| 1083 | ): |
Markus Armbruster | 013b4ef | 2020-03-17 12:54:37 +0100 | [diff] [blame] | 1084 | super().__init__(name, info, doc, ifcond, features) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1085 | self._arg_type_name = arg_type |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 1086 | self.arg_type: Optional[QAPISchemaObjectType] = None |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1087 | self.boxed = boxed |
| 1088 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 1089 | def check(self, schema: QAPISchema) -> None: |
Markus Armbruster | 2cae67b | 2020-03-04 16:59:31 +0100 | [diff] [blame] | 1090 | super().check(schema) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1091 | if self._arg_type_name: |
John Snow | 9bda6c7 | 2024-03-15 16:22:45 +0100 | [diff] [blame] | 1092 | typ = schema.resolve_type( |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1093 | self._arg_type_name, self.info, "event's 'data'") |
John Snow | 9bda6c7 | 2024-03-15 16:22:45 +0100 | [diff] [blame] | 1094 | if not isinstance(typ, QAPISchemaObjectType): |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1095 | raise QAPISemError( |
| 1096 | self.info, |
| 1097 | "event's 'data' cannot take %s" |
John Snow | 9bda6c7 | 2024-03-15 16:22:45 +0100 | [diff] [blame] | 1098 | % typ.describe()) |
| 1099 | self.arg_type = typ |
Markus Armbruster | 3ff2a5a | 2024-03-15 16:33:23 +0100 | [diff] [blame] | 1100 | if self.arg_type.branches and not self.boxed: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1101 | raise QAPISemError( |
| 1102 | self.info, |
| 1103 | "event's 'data' can take %s only with 'boxed': true" |
| 1104 | % self.arg_type.describe()) |
Markus Armbruster | de3b3f5 | 2023-03-16 08:13:25 +0100 | [diff] [blame] | 1105 | self.arg_type.check(schema) |
| 1106 | if self.arg_type.has_conditional_members() and not self.boxed: |
| 1107 | raise QAPISemError( |
| 1108 | self.info, |
| 1109 | "conditional event arguments require 'boxed': true") |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1110 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 1111 | def connect_doc(self, doc: Optional[QAPIDoc] = None) -> None: |
Markus Armbruster | e4405b3 | 2020-03-17 12:54:36 +0100 | [diff] [blame] | 1112 | super().connect_doc(doc) |
Markus Armbruster | bf83f04 | 2019-10-24 13:02:29 +0200 | [diff] [blame] | 1113 | doc = doc or self.doc |
| 1114 | if doc: |
| 1115 | if self.arg_type and self.arg_type.is_implicit(): |
| 1116 | self.arg_type.connect_doc(doc) |
| 1117 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 1118 | def visit(self, visitor: QAPISchemaVisitor) -> None: |
Markus Armbruster | 2cae67b | 2020-03-04 16:59:31 +0100 | [diff] [blame] | 1119 | super().visit(visitor) |
Markus Armbruster | 013b4ef | 2020-03-17 12:54:37 +0100 | [diff] [blame] | 1120 | visitor.visit_event( |
| 1121 | self.name, self.info, self.ifcond, self.features, |
| 1122 | self.arg_type, self.boxed) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1123 | |
| 1124 | |
Markus Armbruster | baa310f | 2020-03-04 16:59:29 +0100 | [diff] [blame] | 1125 | class QAPISchema: |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 1126 | def __init__(self, fname: str): |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1127 | self.fname = fname |
John Snow | 3404e57 | 2021-05-19 14:39:37 -0400 | [diff] [blame] | 1128 | |
| 1129 | try: |
| 1130 | parser = QAPISchemaParser(fname) |
| 1131 | except OSError as err: |
| 1132 | raise QAPIError( |
| 1133 | f"can't read schema file '{fname}': {err.strerror}" |
| 1134 | ) from err |
| 1135 | |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1136 | exprs = check_exprs(parser.exprs) |
| 1137 | self.docs = parser.docs |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 1138 | self._entity_list: List[QAPISchemaEntity] = [] |
| 1139 | self._entity_dict: Dict[str, QAPISchemaDefinition] = {} |
| 1140 | self._module_dict: Dict[str, QAPISchemaModule] = OrderedDict() |
Markus Armbruster | a9f1dd7 | 2019-11-20 19:25:49 +0100 | [diff] [blame] | 1141 | self._schema_dir = os.path.dirname(fname) |
John Snow | 39b2d83 | 2021-02-01 14:37:41 -0500 | [diff] [blame] | 1142 | self._make_module(QAPISchemaModule.BUILTIN_MODULE_NAME) |
Markus Armbruster | a9f1dd7 | 2019-11-20 19:25:49 +0100 | [diff] [blame] | 1143 | self._make_module(fname) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1144 | self._predefining = True |
| 1145 | self._def_predefineds() |
| 1146 | self._predefining = False |
| 1147 | self._def_exprs(exprs) |
| 1148 | self.check() |
| 1149 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 1150 | def _def_entity(self, ent: QAPISchemaEntity) -> None: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1151 | self._entity_list.append(ent) |
John Snow | 2418d1c | 2024-03-15 16:22:41 +0100 | [diff] [blame] | 1152 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 1153 | def _def_definition(self, defn: QAPISchemaDefinition) -> None: |
John Snow | 2418d1c | 2024-03-15 16:22:41 +0100 | [diff] [blame] | 1154 | # Only the predefined types are allowed to not have info |
| 1155 | assert defn.info or self._predefining |
| 1156 | self._def_entity(defn) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1157 | # TODO reject names that differ only in '_' vs. '.' vs. '-', |
| 1158 | # because they're liable to clash in generated C. |
John Snow | 2418d1c | 2024-03-15 16:22:41 +0100 | [diff] [blame] | 1159 | other_defn = self._entity_dict.get(defn.name) |
| 1160 | if other_defn: |
| 1161 | if other_defn.info: |
| 1162 | where = QAPISourceError(other_defn.info, "previous definition") |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1163 | raise QAPISemError( |
John Snow | 2418d1c | 2024-03-15 16:22:41 +0100 | [diff] [blame] | 1164 | defn.info, |
| 1165 | "'%s' is already defined\n%s" % (defn.name, where)) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1166 | raise QAPISemError( |
John Snow | 2418d1c | 2024-03-15 16:22:41 +0100 | [diff] [blame] | 1167 | defn.info, "%s is already defined" % other_defn.describe()) |
| 1168 | self._entity_dict[defn.name] = defn |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1169 | |
John Snow | f64e753 | 2024-06-26 18:21:08 -0400 | [diff] [blame] | 1170 | def lookup_entity(self, name: str) -> Optional[QAPISchemaEntity]: |
Markus Armbruster | 060b5a9 | 2024-03-15 16:23:01 +0100 | [diff] [blame] | 1171 | return self._entity_dict.get(name) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1172 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 1173 | def lookup_type(self, name: str) -> Optional[QAPISchemaType]: |
Markus Armbruster | 060b5a9 | 2024-03-15 16:23:01 +0100 | [diff] [blame] | 1174 | typ = self.lookup_entity(name) |
| 1175 | if isinstance(typ, QAPISchemaType): |
| 1176 | return typ |
| 1177 | return None |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1178 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 1179 | def resolve_type( |
| 1180 | self, |
| 1181 | name: str, |
| 1182 | info: Optional[QAPISourceInfo], |
| 1183 | what: Union[None, str, Callable[[QAPISourceInfo], str]], |
| 1184 | ) -> QAPISchemaType: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1185 | typ = self.lookup_type(name) |
| 1186 | if not typ: |
John Snow | 802a3e3 | 2024-03-15 16:22:47 +0100 | [diff] [blame] | 1187 | assert info and what # built-in types must not fail lookup |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1188 | if callable(what): |
| 1189 | what = what(info) |
| 1190 | raise QAPISemError( |
| 1191 | info, "%s uses unknown type '%s'" % (what, name)) |
| 1192 | return typ |
| 1193 | |
John Snow | e2bbc4e | 2021-02-01 14:37:39 -0500 | [diff] [blame] | 1194 | def _module_name(self, fname: str) -> str: |
John Snow | 98967c2 | 2021-02-01 14:37:36 -0500 | [diff] [blame] | 1195 | if QAPISchemaModule.is_system_module(fname): |
| 1196 | return fname |
Markus Armbruster | a9f1dd7 | 2019-11-20 19:25:49 +0100 | [diff] [blame] | 1197 | return os.path.relpath(fname, self._schema_dir) |
| 1198 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 1199 | def _make_module(self, fname: str) -> QAPISchemaModule: |
Markus Armbruster | a9f1dd7 | 2019-11-20 19:25:49 +0100 | [diff] [blame] | 1200 | name = self._module_name(fname) |
Markus Armbruster | 8ec0e1a | 2020-03-04 16:59:32 +0100 | [diff] [blame] | 1201 | if name not in self._module_dict: |
Markus Armbruster | a9f1dd7 | 2019-11-20 19:25:49 +0100 | [diff] [blame] | 1202 | self._module_dict[name] = QAPISchemaModule(name) |
| 1203 | return self._module_dict[name] |
| 1204 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 1205 | def module_by_fname(self, fname: str) -> QAPISchemaModule: |
Markus Armbruster | a9f1dd7 | 2019-11-20 19:25:49 +0100 | [diff] [blame] | 1206 | name = self._module_name(fname) |
Markus Armbruster | a9f1dd7 | 2019-11-20 19:25:49 +0100 | [diff] [blame] | 1207 | return self._module_dict[name] |
| 1208 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 1209 | def _def_include(self, expr: QAPIExpression) -> None: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1210 | include = expr['include'] |
John Snow | 4201105 | 2023-02-14 19:00:09 -0500 | [diff] [blame] | 1211 | assert expr.doc is None |
| 1212 | self._def_entity( |
| 1213 | QAPISchemaInclude(self._make_module(include), expr.info)) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1214 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 1215 | def _def_builtin_type( |
| 1216 | self, name: str, json_type: str, c_type: str |
| 1217 | ) -> None: |
John Snow | 2418d1c | 2024-03-15 16:22:41 +0100 | [diff] [blame] | 1218 | self._def_definition(QAPISchemaBuiltinType(name, json_type, c_type)) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1219 | # Instantiating only the arrays that are actually used would |
| 1220 | # be nice, but we can't as long as their generated code |
| 1221 | # (qapi-builtin-types.[ch]) may be shared by some other |
| 1222 | # schema. |
| 1223 | self._make_array_type(name, None) |
| 1224 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 1225 | def _def_predefineds(self) -> None: |
John Snow | a7aa64a | 2020-10-09 12:15:34 -0400 | [diff] [blame] | 1226 | for t in [('str', 'string', 'char' + POINTER_SUFFIX), |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1227 | ('number', 'number', 'double'), |
| 1228 | ('int', 'int', 'int64_t'), |
| 1229 | ('int8', 'int', 'int8_t'), |
| 1230 | ('int16', 'int', 'int16_t'), |
| 1231 | ('int32', 'int', 'int32_t'), |
| 1232 | ('int64', 'int', 'int64_t'), |
| 1233 | ('uint8', 'int', 'uint8_t'), |
| 1234 | ('uint16', 'int', 'uint16_t'), |
| 1235 | ('uint32', 'int', 'uint32_t'), |
| 1236 | ('uint64', 'int', 'uint64_t'), |
| 1237 | ('size', 'int', 'uint64_t'), |
| 1238 | ('bool', 'boolean', 'bool'), |
John Snow | a7aa64a | 2020-10-09 12:15:34 -0400 | [diff] [blame] | 1239 | ('any', 'value', 'QObject' + POINTER_SUFFIX), |
| 1240 | ('null', 'null', 'QNull' + POINTER_SUFFIX)]: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1241 | self._def_builtin_type(*t) |
| 1242 | self.the_empty_object_type = QAPISchemaObjectType( |
Markus Armbruster | 013b4ef | 2020-03-17 12:54:37 +0100 | [diff] [blame] | 1243 | 'q_empty', None, None, None, None, None, [], None) |
John Snow | 2418d1c | 2024-03-15 16:22:41 +0100 | [diff] [blame] | 1244 | self._def_definition(self.the_empty_object_type) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1245 | |
| 1246 | qtypes = ['none', 'qnull', 'qnum', 'qstring', 'qdict', 'qlist', |
| 1247 | 'qbool'] |
| 1248 | qtype_values = self._make_enum_members( |
| 1249 | [{'name': n} for n in qtypes], None) |
| 1250 | |
John Snow | 2418d1c | 2024-03-15 16:22:41 +0100 | [diff] [blame] | 1251 | self._def_definition(QAPISchemaEnumType( |
Markus Armbruster | 7b29353 | 2024-09-04 13:18:18 +0200 | [diff] [blame] | 1252 | 'QType', None, None, None, None, qtype_values, None)) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1253 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 1254 | def _make_features( |
| 1255 | self, |
| 1256 | features: Optional[List[Dict[str, Any]]], |
| 1257 | info: Optional[QAPISourceInfo], |
| 1258 | ) -> List[QAPISchemaFeature]: |
Markus Armbruster | ed30f58 | 2020-03-17 12:54:41 +0100 | [diff] [blame] | 1259 | if features is None: |
| 1260 | return [] |
Marc-André Lureau | f17539c | 2021-08-04 12:30:57 +0400 | [diff] [blame] | 1261 | return [QAPISchemaFeature(f['name'], info, |
| 1262 | QAPISchemaIfCond(f.get('if'))) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1263 | for f in features] |
| 1264 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 1265 | def _make_enum_member( |
| 1266 | self, |
| 1267 | name: str, |
| 1268 | ifcond: Optional[Union[str, Dict[str, Any]]], |
| 1269 | features: Optional[List[Dict[str, Any]]], |
| 1270 | info: Optional[QAPISourceInfo], |
| 1271 | ) -> QAPISchemaEnumMember: |
Markus Armbruster | b6c1875 | 2021-10-25 06:24:02 +0200 | [diff] [blame] | 1272 | return QAPISchemaEnumMember(name, info, |
| 1273 | QAPISchemaIfCond(ifcond), |
| 1274 | self._make_features(features, info)) |
| 1275 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 1276 | def _make_enum_members( |
| 1277 | self, values: List[Dict[str, Any]], info: Optional[QAPISourceInfo] |
| 1278 | ) -> List[QAPISchemaEnumMember]: |
Markus Armbruster | b6c1875 | 2021-10-25 06:24:02 +0200 | [diff] [blame] | 1279 | return [self._make_enum_member(v['name'], v.get('if'), |
| 1280 | v.get('features'), info) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1281 | for v in values] |
| 1282 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 1283 | def _make_array_type( |
| 1284 | self, element_type: str, info: Optional[QAPISourceInfo] |
| 1285 | ) -> str: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1286 | name = element_type + 'List' # reserved by check_defn_name_str() |
| 1287 | if not self.lookup_type(name): |
John Snow | 2418d1c | 2024-03-15 16:22:41 +0100 | [diff] [blame] | 1288 | self._def_definition(QAPISchemaArrayType( |
| 1289 | name, info, element_type)) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1290 | return name |
| 1291 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 1292 | def _make_implicit_object_type( |
| 1293 | self, |
| 1294 | name: str, |
| 1295 | info: QAPISourceInfo, |
| 1296 | ifcond: QAPISchemaIfCond, |
| 1297 | role: str, |
| 1298 | members: List[QAPISchemaObjectTypeMember], |
| 1299 | ) -> Optional[str]: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1300 | if not members: |
| 1301 | return None |
| 1302 | # See also QAPISchemaObjectTypeMember.describe() |
| 1303 | name = 'q_obj_%s-%s' % (name, role) |
Markus Armbruster | 99e75d8 | 2024-03-15 16:23:00 +0100 | [diff] [blame] | 1304 | typ = self.lookup_entity(name) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1305 | if typ: |
John Snow | f64e753 | 2024-06-26 18:21:08 -0400 | [diff] [blame] | 1306 | assert isinstance(typ, QAPISchemaObjectType) |
Markus Armbruster | 4e99f4b | 2021-09-17 16:31:32 +0200 | [diff] [blame] | 1307 | # The implicit object type has multiple users. This can |
| 1308 | # only be a duplicate definition, which will be flagged |
| 1309 | # later. |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1310 | else: |
John Snow | 2418d1c | 2024-03-15 16:22:41 +0100 | [diff] [blame] | 1311 | self._def_definition(QAPISchemaObjectType( |
Markus Armbruster | 013b4ef | 2020-03-17 12:54:37 +0100 | [diff] [blame] | 1312 | name, info, None, ifcond, None, None, members, None)) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1313 | return name |
| 1314 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 1315 | def _def_enum_type(self, expr: QAPIExpression) -> None: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1316 | name = expr['enum'] |
| 1317 | data = expr['data'] |
| 1318 | prefix = expr.get('prefix') |
Marc-André Lureau | f17539c | 2021-08-04 12:30:57 +0400 | [diff] [blame] | 1319 | ifcond = QAPISchemaIfCond(expr.get('if')) |
John Snow | 4201105 | 2023-02-14 19:00:09 -0500 | [diff] [blame] | 1320 | info = expr.info |
Markus Armbruster | ed30f58 | 2020-03-17 12:54:41 +0100 | [diff] [blame] | 1321 | features = self._make_features(expr.get('features'), info) |
John Snow | 2418d1c | 2024-03-15 16:22:41 +0100 | [diff] [blame] | 1322 | self._def_definition(QAPISchemaEnumType( |
John Snow | 4201105 | 2023-02-14 19:00:09 -0500 | [diff] [blame] | 1323 | name, info, expr.doc, ifcond, features, |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1324 | self._make_enum_members(data, info), prefix)) |
| 1325 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 1326 | def _make_member( |
| 1327 | self, |
| 1328 | name: str, |
| 1329 | typ: Union[List[str], str], |
| 1330 | ifcond: QAPISchemaIfCond, |
| 1331 | features: Optional[List[Dict[str, Any]]], |
| 1332 | info: QAPISourceInfo, |
| 1333 | ) -> QAPISchemaObjectTypeMember: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1334 | optional = False |
| 1335 | if name.startswith('*'): |
| 1336 | name = name[1:] |
| 1337 | optional = True |
| 1338 | if isinstance(typ, list): |
| 1339 | assert len(typ) == 1 |
| 1340 | typ = self._make_array_type(typ[0], info) |
Markus Armbruster | 84ab008 | 2020-03-17 12:54:45 +0100 | [diff] [blame] | 1341 | return QAPISchemaObjectTypeMember(name, info, typ, optional, ifcond, |
| 1342 | self._make_features(features, info)) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1343 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 1344 | def _make_members( |
| 1345 | self, |
| 1346 | data: Dict[str, Any], |
| 1347 | info: QAPISourceInfo, |
| 1348 | ) -> List[QAPISchemaObjectTypeMember]: |
Marc-André Lureau | f17539c | 2021-08-04 12:30:57 +0400 | [diff] [blame] | 1349 | return [self._make_member(key, value['type'], |
| 1350 | QAPISchemaIfCond(value.get('if')), |
Markus Armbruster | 84ab008 | 2020-03-17 12:54:45 +0100 | [diff] [blame] | 1351 | value.get('features'), info) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1352 | for (key, value) in data.items()] |
| 1353 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 1354 | def _def_struct_type(self, expr: QAPIExpression) -> None: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1355 | name = expr['struct'] |
| 1356 | base = expr.get('base') |
| 1357 | data = expr['data'] |
John Snow | 4201105 | 2023-02-14 19:00:09 -0500 | [diff] [blame] | 1358 | info = expr.info |
Marc-André Lureau | f17539c | 2021-08-04 12:30:57 +0400 | [diff] [blame] | 1359 | ifcond = QAPISchemaIfCond(expr.get('if')) |
Markus Armbruster | ed30f58 | 2020-03-17 12:54:41 +0100 | [diff] [blame] | 1360 | features = self._make_features(expr.get('features'), info) |
John Snow | 2418d1c | 2024-03-15 16:22:41 +0100 | [diff] [blame] | 1361 | self._def_definition(QAPISchemaObjectType( |
John Snow | 4201105 | 2023-02-14 19:00:09 -0500 | [diff] [blame] | 1362 | name, info, expr.doc, ifcond, features, base, |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1363 | self._make_members(data, info), |
Markus Armbruster | 013b4ef | 2020-03-17 12:54:37 +0100 | [diff] [blame] | 1364 | None)) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1365 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 1366 | def _make_variant( |
| 1367 | self, |
| 1368 | case: str, |
| 1369 | typ: str, |
| 1370 | ifcond: QAPISchemaIfCond, |
| 1371 | info: QAPISourceInfo, |
| 1372 | ) -> QAPISchemaVariant: |
Paolo Bonzini | a580694 | 2022-03-21 17:42:41 +0100 | [diff] [blame] | 1373 | if isinstance(typ, list): |
| 1374 | assert len(typ) == 1 |
| 1375 | typ = self._make_array_type(typ[0], info) |
Markus Armbruster | 5858fd1 | 2020-03-17 12:54:43 +0100 | [diff] [blame] | 1376 | return QAPISchemaVariant(case, info, typ, ifcond) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1377 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 1378 | def _def_union_type(self, expr: QAPIExpression) -> None: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1379 | name = expr['union'] |
Markus Armbruster | 4e99f4b | 2021-09-17 16:31:32 +0200 | [diff] [blame] | 1380 | base = expr['base'] |
| 1381 | tag_name = expr['discriminator'] |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1382 | data = expr['data'] |
John Snow | 4201105 | 2023-02-14 19:00:09 -0500 | [diff] [blame] | 1383 | assert isinstance(data, dict) |
| 1384 | info = expr.info |
Marc-André Lureau | f17539c | 2021-08-04 12:30:57 +0400 | [diff] [blame] | 1385 | ifcond = QAPISchemaIfCond(expr.get('if')) |
Markus Armbruster | ed30f58 | 2020-03-17 12:54:41 +0100 | [diff] [blame] | 1386 | features = self._make_features(expr.get('features'), info) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1387 | if isinstance(base, dict): |
| 1388 | base = self._make_implicit_object_type( |
Markus Armbruster | a710e1c | 2019-10-24 13:02:30 +0200 | [diff] [blame] | 1389 | name, info, ifcond, |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1390 | 'base', self._make_members(base, info)) |
Markus Armbruster | 4e99f4b | 2021-09-17 16:31:32 +0200 | [diff] [blame] | 1391 | variants = [ |
| 1392 | self._make_variant(key, value['type'], |
| 1393 | QAPISchemaIfCond(value.get('if')), |
| 1394 | info) |
| 1395 | for (key, value) in data.items()] |
John Snow | 4201105 | 2023-02-14 19:00:09 -0500 | [diff] [blame] | 1396 | members: List[QAPISchemaObjectTypeMember] = [] |
John Snow | 2418d1c | 2024-03-15 16:22:41 +0100 | [diff] [blame] | 1397 | self._def_definition( |
John Snow | 4201105 | 2023-02-14 19:00:09 -0500 | [diff] [blame] | 1398 | QAPISchemaObjectType(name, info, expr.doc, ifcond, features, |
Markus Armbruster | 013b4ef | 2020-03-17 12:54:37 +0100 | [diff] [blame] | 1399 | base, members, |
Markus Armbruster | 1d067e3 | 2024-03-15 20:32:41 +0100 | [diff] [blame] | 1400 | QAPISchemaBranches( |
| 1401 | info, variants, tag_name))) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1402 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 1403 | def _def_alternate_type(self, expr: QAPIExpression) -> None: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1404 | name = expr['alternate'] |
| 1405 | data = expr['data'] |
John Snow | 4201105 | 2023-02-14 19:00:09 -0500 | [diff] [blame] | 1406 | assert isinstance(data, dict) |
Marc-André Lureau | f17539c | 2021-08-04 12:30:57 +0400 | [diff] [blame] | 1407 | ifcond = QAPISchemaIfCond(expr.get('if')) |
John Snow | 4201105 | 2023-02-14 19:00:09 -0500 | [diff] [blame] | 1408 | info = expr.info |
Markus Armbruster | ed30f58 | 2020-03-17 12:54:41 +0100 | [diff] [blame] | 1409 | features = self._make_features(expr.get('features'), info) |
Marc-André Lureau | f17539c | 2021-08-04 12:30:57 +0400 | [diff] [blame] | 1410 | variants = [ |
| 1411 | self._make_variant(key, value['type'], |
| 1412 | QAPISchemaIfCond(value.get('if')), |
| 1413 | info) |
| 1414 | for (key, value) in data.items()] |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1415 | tag_member = QAPISchemaObjectTypeMember('type', info, 'QType', False) |
John Snow | 2418d1c | 2024-03-15 16:22:41 +0100 | [diff] [blame] | 1416 | self._def_definition( |
John Snow | 4201105 | 2023-02-14 19:00:09 -0500 | [diff] [blame] | 1417 | QAPISchemaAlternateType( |
| 1418 | name, info, expr.doc, ifcond, features, |
Markus Armbruster | 1d067e3 | 2024-03-15 20:32:41 +0100 | [diff] [blame] | 1419 | QAPISchemaAlternatives(info, variants, tag_member))) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1420 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 1421 | def _def_command(self, expr: QAPIExpression) -> None: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1422 | name = expr['command'] |
| 1423 | data = expr.get('data') |
| 1424 | rets = expr.get('returns') |
| 1425 | gen = expr.get('gen', True) |
| 1426 | success_response = expr.get('success-response', True) |
| 1427 | boxed = expr.get('boxed', False) |
| 1428 | allow_oob = expr.get('allow-oob', False) |
| 1429 | allow_preconfig = expr.get('allow-preconfig', False) |
Kevin Wolf | 04f2236 | 2020-10-05 17:58:49 +0200 | [diff] [blame] | 1430 | coroutine = expr.get('coroutine', False) |
Marc-André Lureau | f17539c | 2021-08-04 12:30:57 +0400 | [diff] [blame] | 1431 | ifcond = QAPISchemaIfCond(expr.get('if')) |
John Snow | 4201105 | 2023-02-14 19:00:09 -0500 | [diff] [blame] | 1432 | info = expr.info |
Markus Armbruster | ed30f58 | 2020-03-17 12:54:41 +0100 | [diff] [blame] | 1433 | features = self._make_features(expr.get('features'), info) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1434 | if isinstance(data, OrderedDict): |
| 1435 | data = self._make_implicit_object_type( |
Markus Armbruster | 013b4ef | 2020-03-17 12:54:37 +0100 | [diff] [blame] | 1436 | name, info, ifcond, |
| 1437 | 'arg', self._make_members(data, info)) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1438 | if isinstance(rets, list): |
| 1439 | assert len(rets) == 1 |
| 1440 | rets = self._make_array_type(rets[0], info) |
John Snow | 2418d1c | 2024-03-15 16:22:41 +0100 | [diff] [blame] | 1441 | self._def_definition( |
| 1442 | QAPISchemaCommand(name, info, expr.doc, ifcond, features, data, |
| 1443 | rets, gen, success_response, boxed, allow_oob, |
| 1444 | allow_preconfig, coroutine)) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1445 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 1446 | def _def_event(self, expr: QAPIExpression) -> None: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1447 | name = expr['event'] |
| 1448 | data = expr.get('data') |
| 1449 | boxed = expr.get('boxed', False) |
Marc-André Lureau | f17539c | 2021-08-04 12:30:57 +0400 | [diff] [blame] | 1450 | ifcond = QAPISchemaIfCond(expr.get('if')) |
John Snow | 4201105 | 2023-02-14 19:00:09 -0500 | [diff] [blame] | 1451 | info = expr.info |
Markus Armbruster | ed30f58 | 2020-03-17 12:54:41 +0100 | [diff] [blame] | 1452 | features = self._make_features(expr.get('features'), info) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1453 | if isinstance(data, OrderedDict): |
| 1454 | data = self._make_implicit_object_type( |
Markus Armbruster | 013b4ef | 2020-03-17 12:54:37 +0100 | [diff] [blame] | 1455 | name, info, ifcond, |
| 1456 | 'arg', self._make_members(data, info)) |
John Snow | 2418d1c | 2024-03-15 16:22:41 +0100 | [diff] [blame] | 1457 | self._def_definition(QAPISchemaEvent(name, info, expr.doc, ifcond, |
| 1458 | features, data, boxed)) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1459 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 1460 | def _def_exprs(self, exprs: List[QAPIExpression]) -> None: |
John Snow | 4201105 | 2023-02-14 19:00:09 -0500 | [diff] [blame] | 1461 | for expr in exprs: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1462 | if 'enum' in expr: |
John Snow | 4201105 | 2023-02-14 19:00:09 -0500 | [diff] [blame] | 1463 | self._def_enum_type(expr) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1464 | elif 'struct' in expr: |
John Snow | 4201105 | 2023-02-14 19:00:09 -0500 | [diff] [blame] | 1465 | self._def_struct_type(expr) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1466 | elif 'union' in expr: |
John Snow | 4201105 | 2023-02-14 19:00:09 -0500 | [diff] [blame] | 1467 | self._def_union_type(expr) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1468 | elif 'alternate' in expr: |
John Snow | 4201105 | 2023-02-14 19:00:09 -0500 | [diff] [blame] | 1469 | self._def_alternate_type(expr) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1470 | elif 'command' in expr: |
John Snow | 4201105 | 2023-02-14 19:00:09 -0500 | [diff] [blame] | 1471 | self._def_command(expr) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1472 | elif 'event' in expr: |
John Snow | 4201105 | 2023-02-14 19:00:09 -0500 | [diff] [blame] | 1473 | self._def_event(expr) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1474 | elif 'include' in expr: |
John Snow | 4201105 | 2023-02-14 19:00:09 -0500 | [diff] [blame] | 1475 | self._def_include(expr) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1476 | else: |
| 1477 | assert False |
| 1478 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 1479 | def check(self) -> None: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1480 | for ent in self._entity_list: |
| 1481 | ent.check(self) |
Markus Armbruster | ee1e6a1 | 2019-10-24 13:02:26 +0200 | [diff] [blame] | 1482 | ent.connect_doc() |
Markus Armbruster | a9f1dd7 | 2019-11-20 19:25:49 +0100 | [diff] [blame] | 1483 | for ent in self._entity_list: |
| 1484 | ent.set_module(self) |
Markus Armbruster | fedc04c | 2024-02-16 15:58:36 +0100 | [diff] [blame] | 1485 | for doc in self.docs: |
| 1486 | doc.check() |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1487 | |
John Snow | 4ed3fe0 | 2024-03-15 16:22:57 +0100 | [diff] [blame] | 1488 | def visit(self, visitor: QAPISchemaVisitor) -> None: |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1489 | visitor.visit_begin(self) |
Markus Armbruster | 3e7fb58 | 2019-11-20 19:25:50 +0100 | [diff] [blame] | 1490 | for mod in self._module_dict.values(): |
| 1491 | mod.visit(visitor) |
Markus Armbruster | e6c42b9 | 2019-10-18 09:43:44 +0200 | [diff] [blame] | 1492 | visitor.visit_end() |