qemu-devel.nongnu.org archive mirror
 help / color / mirror / Atom feed
From: John Snow <jsnow@redhat.com>
To: Markus Armbruster <armbru@redhat.com>
Cc: Cleber Rosa <crosa@redhat.com>,
	qemu-devel@nongnu.org, Eduardo Habkost <ehabkost@redhat.com>
Subject: Re: [PATCH v5 13/15] qapi/introspect.py: add introspect.json dummy types
Date: Mon, 8 Feb 2021 10:56:37 -0500	[thread overview]
Message-ID: <17c0e654-2aa1-7ab4-3c9b-018dc648b5ca@redhat.com> (raw)
In-Reply-To: <87czxa36q6.fsf@dusky.pond.sub.org>

On 2/8/21 10:43 AM, Markus Armbruster wrote:
> John Snow <jsnow@redhat.com> writes:
> 
>> Add some aliases that declare intent for some of the "dictly-typed"
>> objects we pass around in introspect.py.
>>
>> Signed-off-by: John Snow <jsnow@redhat.com>
>>
>> ---
>>
>> This patch is optional, it can be dropped if desired. If it's taken,
>> it's probably best to squash it with the prior patch.  It is purely for
>> the sake of demonstrating what the _DObject was meant to convey: a
>> Python Dict that represents some JSON Object. It does not add any type
>> safety in and of itself, but does have some (light) annotational
>> benefit. In this case, it's usually a specific data structure from the
>> QAPI Schema we are referencing, but do not have "compile-time" access to
>> in Python.
>>
>> These are loosely typed and don't bother reproducing the exact structure
>> of the real types. Python 3.6 does not have support for TypedDict
>> structures, so this is as good as we can do without involving a
>> third-party library (e.g. Pydantic), in which we might be able to say:
>>
>>
>> class SchemaMetaType(str, enum.Enum):
>>      BUILTIN = "builtin"
>>      ENUM = "enum"
>>      ARRAY = "array"
>>      OBJECT = "object"
>>      ALTERNATE = "alternate"
>>      COMMAND = "command"
>>      EVENT = "event"
>>
>>
>> class SchemaInfo(pydantic.BaseModel):
>>      name: str
>>      meta-type: SchemaMetaType
>>      features: Optional[List[str]]
>>      data: Union[SchemaInfoBuiltin, SchemaInfoEnum, SchemaInfoArray,
>>                  SchemaInfoObject, SchemaInfoAlternate, SchemaInfoCommand,
>>                  SchemaInfoEvent]
>>
>>
>> However, the cost of reproducing and keeping these structure definitions
>> in-sync between the user-defined portion of the schema and the code
>> generator is likely not worth doing any such thing. However, it does
>> illustrate an interesting dependency the generator has on the
>> user-defined schema itself in terms of types.
>>
>> So, I settled on using some light types that suggest the form of the
>> object instead of enforcing the form.
>>
>> Signed-off-by: John Snow <jsnow@redhat.com>
>> ---
>>   scripts/qapi/introspect.py | 51 +++++++++++++++++++++++---------------
>>   1 file changed, 31 insertions(+), 20 deletions(-)
>>
>> diff --git a/scripts/qapi/introspect.py b/scripts/qapi/introspect.py
>> index 3afcdda7446..2a39726f40a 100644
>> --- a/scripts/qapi/introspect.py
>> +++ b/scripts/qapi/introspect.py
>> @@ -66,10 +66,15 @@
>>   _value = Union[_scalar, _nonscalar]
>>   TreeValue = Union[_value, 'Annotated[_value]']
>>   
>> -# This is an alias for an arbitrary JSON object, represented here as a Dict.
>> -# It is stricter on the value type than the recursive definition above.
>> -# It is used to represent SchemaInfo-related structures exclusively.
>> -_DObject = Dict[str, object]
>> +# These types are based on structures defined in QEMU's schema, so we lack
>> +# precise types for them here. Python 3.6 does not offer TypedDict constructs,
>> +# so they are broadly typed here as simple Python Dicts.
>> +SchemaInfo = Dict[str, object]
>> +SchemaInfoObject = Dict[str, object]
>> +SchemaInfoObjectVariant = Dict[str, object]
>> +SchemaInfoObjectMember = Dict[str, object]
>> +SchemaInfoCommand = Dict[str, object]
>> +
>>   
>>   _NodeT = TypeVar('_NodeT', bound=_value)
>>   
>> @@ -162,7 +167,7 @@ def __init__(self, prefix: str, unmask: bool):
>>               ' * QAPI/QMP schema introspection', __doc__)
>>           self._unmask = unmask
>>           self._schema: Optional[QAPISchema] = None
>> -        self._trees: List[Annotated[_DObject]] = []
>> +        self._trees: List[Annotated[SchemaInfo]] = []
>>           self._used_types: List[QAPISchemaType] = []
>>           self._name_map: Dict[str, str] = {}
>>           self._genc.add(mcgen('''
>> @@ -234,9 +239,18 @@ def _gen_features(features: List[QAPISchemaFeature]
>>                         ) -> List[Annotated[str]]:
>>           return [Annotated(f.name, f.ifcond) for f in features]
>>   
>> -    def _gen_tree(self, name: str, mtype: str, obj: _DObject,
>> +    def _gen_tree(self, name: str, mtype: str, obj: Dict[str, object],
>>                     ifcond: List[str],
>>                     features: Optional[List[QAPISchemaFeature]]) -> None:
>> +        """
>> +        Build and append a SchemaInfo object to self._trees.
>> +
>> +        :param name: The entity's name.
>> +        :param mtype: The entity's meta-type.
>> +        :param obj: Additional entity fields, as appropriate for the meta-type.
>> +        :param ifcond: List of conditionals that apply to this entire entity.
>> +        :param features: Optional features field for SchemaInfo.
>> +        """
>>           comment: Optional[str] = None
>>           if mtype not in ('command', 'event', 'builtin', 'array'):
>>               if not self._unmask:
>> @@ -251,8 +265,8 @@ def _gen_tree(self, name: str, mtype: str, obj: _DObject,
>>           self._trees.append(Annotated(obj, ifcond, comment))
>>   
>>       def _gen_member(self, member: QAPISchemaObjectTypeMember
>> -                    ) -> Annotated[_DObject]:
>> -        obj: _DObject = {
>> +                    ) -> Annotated[SchemaInfoObjectMember]:
>> +        obj: SchemaInfoObjectMember = {
>>               'name': member.name,
>>               'type': self._use_type(member.type)
>>           }
>> @@ -262,13 +276,9 @@ def _gen_member(self, member: QAPISchemaObjectTypeMember
>>               obj['features'] = self._gen_features(member.features)
>>           return Annotated(obj, member.ifcond)
>>   
>> -    def _gen_variants(self, tag_name: str,
>> -                      variants: List[QAPISchemaVariant]) -> _DObject:
>> -        return {'tag': tag_name,
>> -                'variants': [self._gen_variant(v) for v in variants]}
>> -
>> -    def _gen_variant(self, variant: QAPISchemaVariant) -> Annotated[_DObject]:
>> -        obj: _DObject = {
>> +    def _gen_variant(self, variant: QAPISchemaVariant
>> +                     ) -> Annotated[SchemaInfoObjectVariant]:
>> +        obj: SchemaInfoObjectVariant = {
>>               'case': variant.name,
>>               'type': self._use_type(variant.type)
>>           }
>> @@ -300,11 +310,12 @@ def visit_object_type_flat(self, name: str, info: Optional[QAPISourceInfo],
>>                                  features: List[QAPISchemaFeature],
>>                                  members: List[QAPISchemaObjectTypeMember],
>>                                  variants: Optional[QAPISchemaVariants]) -> None:
>> -        obj: _DObject = {'members': [self._gen_member(m) for m in members]}
>> +        obj: SchemaInfoObject = {
>> +            'members': [self._gen_member(m) for m in members]
>> +        }
>>           if variants:
>> -            obj.update(self._gen_variants(variants.tag_member.name,
>> -                                          variants.variants))
>> -
>> +            obj['tag'] = variants.tag_member.name
>> +            obj['variants'] = [self._gen_variant(v) for v in variants.variants]
>>           self._gen_tree(name, 'object', obj, ifcond, features)
>>   
>>       def visit_alternate_type(self, name: str, info: Optional[QAPISourceInfo],
>> @@ -329,7 +340,7 @@ def visit_command(self, name: str, info: Optional[QAPISourceInfo],
>>   
>>           arg_type = arg_type or self._schema.the_empty_object_type
>>           ret_type = ret_type or self._schema.the_empty_object_type
>> -        obj: _DObject = {
>> +        obj: SchemaInfoCommand = {
>>               'arg-type': self._use_type(arg_type),
>>               'ret-type': self._use_type(ret_type)
>>           }
> 
> I see what you mean by "some (light) annotational benefit".
> 
> I figure some of is simply due to going from _DObject, a name that tells
> me nothing, to SchemaInfo, which makes me go aha!
> 

Naming is hard! Using the *exact* names seems helpful.

> The remainder is having the subtypes of SchemaInfo, too. albeit without
> actual type checking.  Worthwhile?
> 

For referential purposes it seems useful, at least in so far as 
declaring intent. When I was trying to type this series, I was not aware 
this module was building structures defined in introspect.json until 
quite late into the series.

I kept wondering: "What are the valid keys here? Who consumes this?" -- 
It wasn't obvious.

I'm leaning towards just keeping them all for the documentation benefit, 
though I admit it *could* mislead someone into believing it's type 
checked when it isn't, really. I tried to call that out with the comment 
at the top, which I hope suffices to diffuse that potential 
faith-based-landmine.

> I agree it should be squashed if we want it, or parts of it.
> 

Squash all of it, IMO.

--js



  reply	other threads:[~2021-02-08 21:39 UTC|newest]

Thread overview: 27+ messages / expand[flat|nested]  mbox.gz  Atom feed  top
2021-02-04  0:31 [PATCH v5 00/15] qapi: static typing conversion, pt2 John Snow
2021-02-04  0:31 ` [PATCH v5 01/15] qapi/introspect.py: assert schema is not None John Snow
2021-02-04  0:31 ` [PATCH v5 02/15] qapi/introspect.py: use _make_tree for features nodes John Snow
2021-02-04  0:31 ` [PATCH v5 03/15] qapi/introspect.py: add _gen_features helper John Snow
2021-02-04  0:31 ` [PATCH v5 04/15] qapi/introspect.py: guard against ifcond/comment misuse John Snow
2021-02-04  0:31 ` [PATCH v5 05/15] qapi/introspect.py: Unify return type of _make_tree() John Snow
2021-02-04  0:31 ` [PATCH v5 06/15] qapi/introspect.py: replace 'extra' dict with 'comment' argument John Snow
2021-02-04  0:31 ` [PATCH v5 07/15] qapi/introspect.py: Always define all 'extra' dict keys John Snow
2021-02-04  0:32 ` [PATCH v5 08/15] qapi/introspect.py: Introduce preliminary tree typing John Snow
2021-02-08 14:29   ` Markus Armbruster
2021-02-04  0:32 ` [PATCH v5 09/15] qapi/introspect.py: create a typed 'Annotated' data strutcure John Snow
2021-02-08 14:36   ` Markus Armbruster
2021-02-08 20:53     ` John Snow
2021-02-04  0:32 ` [PATCH v5 10/15] qapi/introspect.py: improve _tree_to_qlit error message John Snow
2021-02-04  0:32 ` [PATCH v5 11/15] qapi/introspect.py: improve readability of _tree_to_qlit John Snow
2021-02-04  0:32 ` [PATCH v5 12/15] qapi/introspect.py: add type hint annotations John Snow
2021-02-08 15:34   ` Markus Armbruster
2021-02-08 20:45     ` John Snow
2021-02-09  9:17       ` Markus Armbruster
2021-02-10 17:05         ` John Snow
2021-02-04  0:32 ` [PATCH v5 13/15] qapi/introspect.py: add introspect.json dummy types John Snow
2021-02-08 15:43   ` Markus Armbruster
2021-02-08 15:56     ` John Snow [this message]
2021-02-04  0:32 ` [PATCH v5 14/15] qapi/introspect.py: Add docstring to _tree_to_qlit John Snow
2021-02-08 15:45   ` Markus Armbruster
2021-02-08 15:57     ` John Snow
2021-02-04  0:32 ` [PATCH v5 15/15] qapi/introspect.py: Update copyright and authors list John Snow

Reply instructions:

You may reply publicly to this message via plain-text email
using any one of the following methods:

* Save the following mbox file, import it into your mail client,
  and reply-to-all from there: mbox

  Avoid top-posting and favor interleaved quoting:
  https://en.wikipedia.org/wiki/Posting_style#Interleaved_style

* Reply using the --to, --cc, and --in-reply-to
  switches of git-send-email(1):

  git send-email \
    --in-reply-to=17c0e654-2aa1-7ab4-3c9b-018dc648b5ca@redhat.com \
    --to=jsnow@redhat.com \
    --cc=armbru@redhat.com \
    --cc=crosa@redhat.com \
    --cc=ehabkost@redhat.com \
    --cc=qemu-devel@nongnu.org \
    /path/to/YOUR_REPLY

  https://kernel.org/pub/software/scm/git/docs/git-send-email.html

* If your mail client supports setting the In-Reply-To header
  via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).