From nobody Sun Apr  6 00:43:21 2025
Delivered-To: importer@patchew.org
Authentication-Results: mx.zohomail.com;
	dkim=pass;
	spf=pass (zohomail.com: domain of gnu.org designates 209.51.188.17 as
 permitted sender)
  smtp.mailfrom=qemu-devel-bounces+importer=patchew.org@nongnu.org;
	dmarc=pass(p=none dis=none)  header.from=redhat.com
ARC-Seal: i=1; a=rsa-sha256; t=1740368396; cv=none;
	d=zohomail.com; s=zohoarc;
	b=HvPJ1f4gpCuPt/ya462vOgx3FlIxTQaEroATDBJYac5yY8QwyZJsLoeFlIIUSa85VlvK2Uz4qxH5EUwa+CoU/IDH/NgMFJnRjUwmXLjPlk80Tp3DRZTTV7eIzfiv7JJVp4nJmHxjU1zzlx21jC54+nx4LHcEp7k4f9Lk+31vJac=
ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=zohomail.com;
 s=zohoarc;
	t=1740368396;
 h=Content-Transfer-Encoding:Cc:Cc:Date:Date:From:From:In-Reply-To:List-Subscribe:List-Post:List-Id:List-Archive:List-Help:List-Unsubscribe:MIME-Version:Message-ID:References:Sender:Subject:Subject:To:To:Message-Id:Reply-To;
	bh=zMLbHhizeluJkqGGXl5VHpMeg84p0Ut5PhJj+s5R06k=;
	b=KXcESaQUeYx2jQHM0inU5xs9czT/tvFkIT0wtroZ5qVD9HM2dCB6jBW3VZl0fIltaRS0W4nHhKsQbOjXKyA7a44sgL3fGvHMoFZtvI7/vFpi3ojkGGemEZRCQgTnREHHhnBBhwz1zmDwWlK6iLxksfHUvFnd9c4h81ks9rjshpM=
ARC-Authentication-Results: i=1; mx.zohomail.com;
	dkim=pass;
	spf=pass (zohomail.com: domain of gnu.org designates 209.51.188.17 as
 permitted sender)
  smtp.mailfrom=qemu-devel-bounces+importer=patchew.org@nongnu.org;
	dmarc=pass header.from=<jsnow@redhat.com> (p=none dis=none)
Return-Path: <qemu-devel-bounces+importer=patchew.org@nongnu.org>
Received: from lists.gnu.org (lists.gnu.org [209.51.188.17]) by
 mx.zohomail.com
	with SMTPS id 1740368396732107.7121538787884;
 Sun, 23 Feb 2025 19:39:56 -0800 (PST)
Received: from localhost ([::1] helo=lists1p.gnu.org)
	by lists.gnu.org with esmtp (Exim 4.90_1)
	(envelope-from <qemu-devel-bounces@nongnu.org>)
	id 1tmPIx-0001wB-He; Sun, 23 Feb 2025 22:38:51 -0500
Received: from eggs.gnu.org ([2001:470:142:3::10])
 by lists.gnu.org with esmtps (TLS1.2:ECDHE_RSA_AES_256_GCM_SHA384:256)
 (Exim 4.90_1) (envelope-from <jsnow@redhat.com>) id 1tmPIJ-0001Pm-C5
 for qemu-devel@nongnu.org; Sun, 23 Feb 2025 22:38:13 -0500
Received: from us-smtp-delivery-124.mimecast.com ([170.10.133.124])
 by eggs.gnu.org with esmtps (TLS1.2:ECDHE_RSA_AES_256_GCM_SHA384:256)
 (Exim 4.90_1) (envelope-from <jsnow@redhat.com>) id 1tmPIF-0005Rb-TK
 for qemu-devel@nongnu.org; Sun, 23 Feb 2025 22:38:09 -0500
Received: from mx-prod-mc-04.mail-002.prod.us-west-2.aws.redhat.com
 (ec2-54-186-198-63.us-west-2.compute.amazonaws.com [54.186.198.63]) by
 relay.mimecast.com with ESMTP with STARTTLS (version=TLSv1.3,
 cipher=TLS_AES_256_GCM_SHA384) id us-mta-610-CkQb5kcPMEyHKr9havTcwA-1; Sun,
 23 Feb 2025 22:38:03 -0500
Received: from mx-prod-int-05.mail-002.prod.us-west-2.aws.redhat.com
 (mx-prod-int-05.mail-002.prod.us-west-2.aws.redhat.com [10.30.177.17])
 (using TLSv1.3 with cipher TLS_AES_256_GCM_SHA384 (256/256 bits)
 key-exchange X25519 server-signature RSA-PSS (2048 bits) server-digest
 SHA256)
 (No client certificate requested)
 by mx-prod-mc-04.mail-002.prod.us-west-2.aws.redhat.com (Postfix) with ESMTPS
 id 95F0019357B1; Mon, 24 Feb 2025 03:38:01 +0000 (UTC)
Received: from jsnow-thinkpadp16vgen1.westford.csb (unknown [10.22.64.88])
 by mx-prod-int-05.mail-002.prod.us-west-2.aws.redhat.com (Postfix) with ESMTP
 id C810519560A3; Mon, 24 Feb 2025 03:37:59 +0000 (UTC)
DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=redhat.com;
 s=mimecast20190719; t=1740368286;
 h=from:from:reply-to:subject:subject:date:date:message-id:message-id:
 to:to:cc:cc:mime-version:mime-version:
 content-transfer-encoding:content-transfer-encoding:
 in-reply-to:in-reply-to:references:references;
 bh=zMLbHhizeluJkqGGXl5VHpMeg84p0Ut5PhJj+s5R06k=;
 b=fOIJC/s73M0ECm2r0d7LjBYm1nXfUhGRyLClEfmP6Z75/IfgxaRHchKkg/0YWQhKUPgWja
 0V3WJG3dYmGSY4lqLLk6toDrDAbw2nErCzKtU0CWAGRGoBDbWJWZeVFBC0sHpNdYevO7IL
 Lil3PVwfx9BSlER5Ug/QGY76rAk8CWM=
X-MC-Unique: CkQb5kcPMEyHKr9havTcwA-1
X-Mimecast-MFC-AGG-ID: CkQb5kcPMEyHKr9havTcwA_1740368281
From: John Snow <jsnow@redhat.com>
To: qemu-devel@nongnu.org
Cc: Michael Roth <michael.roth@amd.com>, Thomas Huth <thuth@redhat.com>,
 Peter Maydell <peter.maydell@linaro.org>,
 Markus Armbruster <armbru@redhat.com>,
 =?UTF-8?q?Alex=20Benn=C3=A9e?= <alex.bennee@linaro.org>,
 Cleber Rosa <crosa@redhat.com>,
 =?UTF-8?q?Philippe=20Mathieu-Daud=C3=A9?= <philmd@linaro.org>,
 John Snow <jsnow@redhat.com>,
 =?UTF-8?q?Daniel=20P=2E=20Berrang=C3=A9?= <berrange@redhat.com>
Subject: [PATCH 07/10] qapi: expand tags to all doc sections
Date: Sun, 23 Feb 2025 22:37:38 -0500
Message-ID: <20250224033741.222749-8-jsnow@redhat.com>
In-Reply-To: <20250224033741.222749-1-jsnow@redhat.com>
References: <20250224033741.222749-1-jsnow@redhat.com>
MIME-Version: 1.0
Content-Transfer-Encoding: quoted-printable
X-Scanned-By: MIMEDefang 3.0 on 10.30.177.17
Received-SPF: pass (zohomail.com: domain of gnu.org designates 209.51.188.17
 as permitted sender) client-ip=209.51.188.17;
 envelope-from=qemu-devel-bounces+importer=patchew.org@nongnu.org;
 helo=lists.gnu.org;
Received-SPF: pass client-ip=170.10.133.124; envelope-from=jsnow@redhat.com;
 helo=us-smtp-delivery-124.mimecast.com
X-Spam_score_int: -25
X-Spam_score: -2.6
X-Spam_bar: --
X-Spam_report: (-2.6 / 5.0 requ) BAYES_00=-1.9, DKIMWL_WL_HIGH=-0.446,
 DKIM_SIGNED=0.1, DKIM_VALID=-0.1, DKIM_VALID_AU=-0.1, DKIM_VALID_EF=-0.1,
 RCVD_IN_DNSWL_NONE=-0.0001, RCVD_IN_MSPIKE_H2=-0.01,
 RCVD_IN_VALIDITY_RPBL_BLOCKED=0.001, RCVD_IN_VALIDITY_SAFE_BLOCKED=0.001,
 SPF_HELO_NONE=0.001, SPF_PASS=-0.001 autolearn=ham autolearn_force=no
X-Spam_action: no action
X-BeenThere: qemu-devel@nongnu.org
X-Mailman-Version: 2.1.29
Precedence: list
List-Id: <qemu-devel.nongnu.org>
List-Unsubscribe: <https://lists.nongnu.org/mailman/options/qemu-devel>,
 <mailto:qemu-devel-request@nongnu.org?subject=unsubscribe>
List-Archive: <https://lists.nongnu.org/archive/html/qemu-devel>
List-Post: <mailto:qemu-devel@nongnu.org>
List-Help: <mailto:qemu-devel-request@nongnu.org?subject=help>
List-Subscribe: <https://lists.nongnu.org/mailman/listinfo/qemu-devel>,
 <mailto:qemu-devel-request@nongnu.org?subject=subscribe>
Errors-To: qemu-devel-bounces+importer=patchew.org@nongnu.org
Sender: qemu-devel-bounces+importer=patchew.org@nongnu.org
X-ZohoMail-DKIM: pass (identity @redhat.com)
X-ZM-MESSAGEID: 1740368399799019100
Content-Type: text/plain; charset="utf-8"

This patch adds an explicit section "kind" to all QAPIDoc
sections. Members/Features are now explicitly marked as such, with the
name now being stored in a dedicated "name" field (which qapidoc.py was
not actually using anyway.)

The qapi-schema tests are updated to account for the new section names;
mostly "TODO" becomes "Todo" and `None` becomes "Plain".

Signed-off-by: John Snow <jsnow@redhat.com>
---
 docs/sphinx/qapidoc.py         |   7 ++-
 scripts/qapi/parser.py         | 109 ++++++++++++++++++++++++---------
 tests/qapi-schema/doc-good.out |  10 +--
 tests/qapi-schema/test-qapi.py |   2 +-
 4 files changed, 90 insertions(+), 38 deletions(-)

diff --git a/docs/sphinx/qapidoc.py b/docs/sphinx/qapidoc.py
index 61997fd21af..d622398f1da 100644
--- a/docs/sphinx/qapidoc.py
+++ b/docs/sphinx/qapidoc.py
@@ -35,6 +35,7 @@
 from docutils.statemachine import ViewList
 from qapi.error import QAPIError, QAPISemError
 from qapi.gen import QAPISchemaVisitor
+from qapi.parser import QAPIDoc
 from qapi.schema import QAPISchema
=20
 from sphinx import addnodes
@@ -258,11 +259,11 @@ def _nodes_for_sections(self, doc):
         """Return list of doctree nodes for additional sections"""
         nodelist =3D []
         for section in doc.sections:
-            if section.tag and section.tag =3D=3D 'TODO':
+            if section.kind =3D=3D QAPIDoc.Kind.TODO:
                 # Hide TODO: sections
                 continue
=20
-            if not section.tag:
+            if section.kind =3D=3D QAPIDoc.Kind.PLAIN:
                 # Sphinx cannot handle sectionless titles;
                 # Instead, just append the results to the prior section.
                 container =3D nodes.container()
@@ -270,7 +271,7 @@ def _nodes_for_sections(self, doc):
                 nodelist +=3D container.children
                 continue
=20
-            snode =3D self._make_section(section.tag)
+            snode =3D self._make_section(section.kind.name.title())
             self._parse_text_into_node(dedent(section.text), snode)
             nodelist.append(snode)
         return nodelist
diff --git a/scripts/qapi/parser.py b/scripts/qapi/parser.py
index 36cb64a677a..c3004aa70c6 100644
--- a/scripts/qapi/parser.py
+++ b/scripts/qapi/parser.py
@@ -15,6 +15,7 @@
 # See the COPYING file in the top-level directory.
=20
 from collections import OrderedDict
+import enum
 import os
 import re
 from typing import (
@@ -575,7 +576,10 @@ def get_doc(self) -> 'QAPIDoc':
                         )
                         raise QAPIParseError(self, emsg)
=20
-                    doc.new_tagged_section(self.info, match.group(1))
+                    doc.new_tagged_section(
+                        self.info,
+                        QAPIDoc.Kind.from_string(match.group(1))
+                    )
                     text =3D line[match.end():]
                     if text:
                         doc.append_line(text)
@@ -586,7 +590,7 @@ def get_doc(self) -> 'QAPIDoc':
                         self,
                         "unexpected '=3D' markup in definition documentati=
on")
                 else:
-                    # tag-less paragraph
+                    # plain paragraph(s)
                     doc.ensure_untagged_section(self.info)
                     doc.append_line(line)
                     line =3D self.get_doc_paragraph(doc)
@@ -635,14 +639,37 @@ class QAPIDoc:
     Free-form documentation blocks consist only of a body section.
     """
=20
+    class Kind(enum.Enum):
+        PLAIN =3D 0
+        MEMBER =3D 1
+        FEATURE =3D 2
+        RETURNS =3D 3
+        ERRORS =3D 4
+        SINCE =3D 5
+        TODO =3D 6
+
+        @staticmethod
+        def from_string(kind: str) -> 'QAPIDoc.Kind':
+            return QAPIDoc.Kind[kind.upper()]
+
+        def text_required(self) -> bool:
+            # Only "plain" sections can be empty
+            return self.value not in (0,)
+
+        def __str__(self) -> str:
+            return self.name.title()
+
     class Section:
         # pylint: disable=3Dtoo-few-public-methods
-        def __init__(self, info: QAPISourceInfo,
-                     tag: Optional[str] =3D None):
+        def __init__(
+            self,
+            info: QAPISourceInfo,
+            kind: 'QAPIDoc.Kind',
+        ):
             # section source info, i.e. where it begins
             self.info =3D info
-            # section tag, if any ('Returns', '@name', ...)
-            self.tag =3D tag
+            # section kind
+            self.kind =3D kind
             # section text without tag
             self.text =3D ''
=20
@@ -650,8 +677,14 @@ def append_line(self, line: str) -> None:
             self.text +=3D line + '\n'
=20
     class ArgSection(Section):
-        def __init__(self, info: QAPISourceInfo, tag: str):
-            super().__init__(info, tag)
+        def __init__(
+            self,
+            info: QAPISourceInfo,
+            kind: 'QAPIDoc.Kind',
+            name: str
+        ):
+            super().__init__(info, kind)
+            self.name =3D name
             self.member: Optional['QAPISchemaMember'] =3D None
=20
         def connect(self, member: 'QAPISchemaMember') -> None:
@@ -663,7 +696,9 @@ def __init__(self, info: QAPISourceInfo, symbol: Option=
al[str] =3D None):
         # definition doc's symbol, None for free-form doc
         self.symbol: Optional[str] =3D symbol
         # the sections in textual order
-        self.all_sections: List[QAPIDoc.Section] =3D [QAPIDoc.Section(info=
)]
+        self.all_sections: List[QAPIDoc.Section] =3D [
+            QAPIDoc.Section(info, QAPIDoc.Kind.PLAIN)
+        ]
         # the body section
         self.body: Optional[QAPIDoc.Section] =3D self.all_sections[0]
         # dicts mapping parameter/feature names to their description
@@ -680,12 +715,17 @@ def __init__(self, info: QAPISourceInfo, symbol: Opti=
onal[str] =3D None):
     def end(self) -> None:
         for section in self.all_sections:
             section.text =3D section.text.strip('\n')
-            if section.tag is not None and section.text =3D=3D '':
+            if section.kind.text_required() and section.text =3D=3D '':
                 raise QAPISemError(
-                    section.info, "text required after '%s:'" % section.ta=
g)
+                    section.info, "text required after '%s:'" % section.ki=
nd)
=20
-    def ensure_untagged_section(self, info: QAPISourceInfo) -> None:
-        if self.all_sections and not self.all_sections[-1].tag:
+    def ensure_untagged_section(
+        self,
+        info: QAPISourceInfo,
+    ) -> None:
+        kind =3D QAPIDoc.Kind.PLAIN
+
+        if self.all_sections and self.all_sections[-1].kind =3D=3D kind:
             # extend current section
             section =3D self.all_sections[-1]
             if not section.text:
@@ -693,46 +733,56 @@ def ensure_untagged_section(self, info: QAPISourceInf=
o) -> None:
                 section.info =3D info
             section.text +=3D '\n'
             return
+
         # start new section
-        section =3D self.Section(info)
+        section =3D self.Section(info, kind)
         self.sections.append(section)
         self.all_sections.append(section)
=20
-    def new_tagged_section(self, info: QAPISourceInfo, tag: str) -> None:
-        section =3D self.Section(info, tag)
-        if tag =3D=3D 'Returns':
+    def new_tagged_section(
+        self,
+        info: QAPISourceInfo,
+        kind: 'QAPIDoc.Kind',
+    ) -> None:
+        section =3D self.Section(info, kind)
+        if kind =3D=3D QAPIDoc.Kind.RETURNS:
             if self.returns:
                 raise QAPISemError(
-                    info, "duplicated '%s' section" % tag)
+                    info, "duplicated '%s' section" % kind)
             self.returns =3D section
-        elif tag =3D=3D 'Errors':
+        elif kind =3D=3D QAPIDoc.Kind.ERRORS:
             if self.errors:
                 raise QAPISemError(
-                    info, "duplicated '%s' section" % tag)
+                    info, "duplicated '%s' section" % kind)
             self.errors =3D section
-        elif tag =3D=3D 'Since':
+        elif kind =3D=3D QAPIDoc.Kind.SINCE:
             if self.since:
                 raise QAPISemError(
-                    info, "duplicated '%s' section" % tag)
+                    info, "duplicated '%s' section" % kind)
             self.since =3D section
         self.sections.append(section)
         self.all_sections.append(section)
=20
-    def _new_description(self, info: QAPISourceInfo, name: str,
-                         desc: Dict[str, ArgSection]) -> None:
+    def _new_description(
+        self,
+        info: QAPISourceInfo,
+        name: str,
+        kind: 'QAPIDoc.Kind',
+        desc: Dict[str, ArgSection]
+    ) -> None:
         if not name:
             raise QAPISemError(info, "invalid parameter name")
         if name in desc:
             raise QAPISemError(info, "'%s' parameter name duplicated" % na=
me)
-        section =3D self.ArgSection(info, '@' + name)
+        section =3D self.ArgSection(info, kind, name)
         self.all_sections.append(section)
         desc[name] =3D section
=20
     def new_argument(self, info: QAPISourceInfo, name: str) -> None:
-        self._new_description(info, name, self.args)
+        self._new_description(info, name, QAPIDoc.Kind.MEMBER, self.args)
=20
     def new_feature(self, info: QAPISourceInfo, name: str) -> None:
-        self._new_description(info, name, self.features)
+        self._new_description(info, name, QAPIDoc.Kind.FEATURE, self.featu=
res)
=20
     def append_line(self, line: str) -> None:
         self.all_sections[-1].append_line(line)
@@ -744,8 +794,9 @@ def connect_member(self, member: 'QAPISchemaMember') ->=
 None:
                 raise QAPISemError(member.info,
                                    "%s '%s' lacks documentation"
                                    % (member.role, member.name))
-            self.args[member.name] =3D QAPIDoc.ArgSection(
-                self.info, '@' + member.name)
+            section =3D QAPIDoc.ArgSection(
+                self.info, QAPIDoc.Kind.MEMBER, member.name)
+            self.args[member.name] =3D section
         self.args[member.name].connect(member)
=20
     def connect_feature(self, feature: 'QAPISchemaFeature') -> None:
diff --git a/tests/qapi-schema/doc-good.out b/tests/qapi-schema/doc-good.out
index ec277be91e9..2d33a305ee7 100644
--- a/tests/qapi-schema/doc-good.out
+++ b/tests/qapi-schema/doc-good.out
@@ -110,7 +110,7 @@ The _one_ {and only}, description on the same line
 Also _one_ {and only}
     feature=3Denum-member-feat
 a member feature
-    section=3DNone
+    section=3DPlain
 @two is undocumented
 doc symbol=3DBase
     body=3D
@@ -168,15 +168,15 @@ description starts on the same line
 a feature
     feature=3Dcmd-feat2
 another feature
-    section=3DNone
+    section=3DPlain
 .. note:: @arg3 is undocumented
     section=3DReturns
 @Object
     section=3DErrors
 some
-    section=3DTODO
+    section=3DTodo
 frobnicate
-    section=3DNone
+    section=3DPlain
 .. admonition:: Notes
=20
  - Lorem ipsum dolor sit amet
@@ -209,7 +209,7 @@ If you're bored enough to read this, go see a video of =
boxed cats
 a feature
     feature=3Dcmd-feat2
 another feature
-    section=3DNone
+    section=3DPlain
 .. qmp-example::
=20
    -> "this example"
diff --git a/tests/qapi-schema/test-qapi.py b/tests/qapi-schema/test-qapi.py
index 7e3f9f4aa1f..bca924309be 100755
--- a/tests/qapi-schema/test-qapi.py
+++ b/tests/qapi-schema/test-qapi.py
@@ -131,7 +131,7 @@ def test_frontend(fname):
         for feat, section in doc.features.items():
             print('    feature=3D%s\n%s' % (feat, section.text))
         for section in doc.sections:
-            print('    section=3D%s\n%s' % (section.tag, section.text))
+            print('    section=3D%s\n%s' % (section.kind, section.text))
=20
=20
 def open_test_result(dir_name, file_name, update):
--=20
2.48.1