[PATCH v3 0/8] qapi-go: add generator for Golang interfaces

Victor Toso posted 8 patches 2 months, 3 weeks ago
docs/devel/index-build.rst          |    1 +
docs/devel/qapi-golang-code-gen.rst |  548 +++++++++
scripts/qapi/golang.py              | 1645 +++++++++++++++++++++++++++
scripts/qapi/main.py                |    3 +
4 files changed, 2197 insertions(+)
create mode 100644 docs/devel/qapi-golang-code-gen.rst
create mode 100644 scripts/qapi/golang.py
[PATCH v3 0/8] qapi-go: add generator for Golang interfaces
Posted by Victor Toso 2 months, 3 weeks ago
This patch series intent is to introduce a generator that produces a Go
module for Go applications to interact over QMP with QEMU.

The initial Goal is to have a Go module that works as intended and can
be improved upon. I'd consider initial releases to be alpha while we
work with utilities tools and libraries on top of this.

The generated code should reside in a separated Git repository, similar
to python-qemu-qmp.

Applications should be able to consume this under qemu.org
namespace (e.g: import "qemu.org/go/qemu"), see Daniel's suggestion:
https://lists.gnu.org/archive/html/qemu-devel/2023-09/msg07024.html

This is the third iteration:
v2: https://lists.gnu.org/archive/html/qemu-devel/2023-10/msg04785.html

I've pushed this series in my gitlab fork:
https://gitlab.com/victortoso/qapi-go/

The fork contains some tests, including tests that were generated from
QAPI's own examples from another generator created for testing, if you
are interested in it:
https://lists.gnu.org/archive/html/qemu-devel/2023-09/msg04946.html

I've generated the qapi-go module over each commit of this series, see:
https://gitlab.com/victortoso/qapi-go/-/commits/qapi-golang-v3-by-patch

I've also generated the qapi-go module over QEMU tags: v9.1.0, v9.2.0:
https://gitlab.com/victortoso/qapi-go/-/commits/qapi-golang-v3-by-tags

--

Sorry that its been awhile between v2 and v3, I had to prioritize other
things. I hope to get this back on track in 2025.

Cheers,
Victor

* Changes:

On generated go:
 - the output should be formatted as gofmt/goimports tools (Daniel)

 - Included QAPI's documentation too (Daniel), see:
   https://lists.gnu.org/archive/html/qemu-devel/2024-11/msg01621.html
    
 - Commands and Events should Marshal directly (Andrea)

On python script:
 - rebased: now uses QAPISchemaBranches, QAPISchemaAlternatives

 - use textwrap as much as possible (Andrea)

 - lots of changes to make the output like gofmt does

Victor Toso (8):
  qapi: golang: Generate enum type
  qapi: golang: Generate alternate types
  qapi: golang: Generate struct types
  qapi: golang: structs: Address nullable members
  qapi: golang: Generate union type
  qapi: golang: Generate event type
  qapi: golang: Generate command type
  docs: add notes on Golang code generator

 docs/devel/index-build.rst          |    1 +
 docs/devel/qapi-golang-code-gen.rst |  548 +++++++++
 scripts/qapi/golang.py              | 1645 +++++++++++++++++++++++++++
 scripts/qapi/main.py                |    3 +
 4 files changed, 2197 insertions(+)
 create mode 100644 docs/devel/qapi-golang-code-gen.rst
 create mode 100644 scripts/qapi/golang.py

-- 
2.47.1
Re: [PATCH v3 0/8] qapi-go: add generator for Golang interfaces
Posted by Markus Armbruster 2 months, 3 weeks ago
Victor Toso <victortoso@redhat.com> writes:

> This patch series intent is to introduce a generator that produces a Go
> module for Go applications to interact over QMP with QEMU.
>
> The initial Goal is to have a Go module that works as intended and can
> be improved upon. I'd consider initial releases to be alpha while we
> work with utilities tools and libraries on top of this.
>
> The generated code should reside in a separated Git repository, similar
> to python-qemu-qmp.
>
> Applications should be able to consume this under qemu.org
> namespace (e.g: import "qemu.org/go/qemu"), see Daniel's suggestion:
> https://lists.gnu.org/archive/html/qemu-devel/2023-09/msg07024.html
>
> This is the third iteration:
> v2: https://lists.gnu.org/archive/html/qemu-devel/2023-10/msg04785.html
>
> I've pushed this series in my gitlab fork:
> https://gitlab.com/victortoso/qapi-go/
>
> The fork contains some tests, including tests that were generated from
> QAPI's own examples from another generator created for testing, if you
> are interested in it:
> https://lists.gnu.org/archive/html/qemu-devel/2023-09/msg04946.html
>
> I've generated the qapi-go module over each commit of this series, see:
> https://gitlab.com/victortoso/qapi-go/-/commits/qapi-golang-v3-by-patch
>
> I've also generated the qapi-go module over QEMU tags: v9.1.0, v9.2.0:
> https://gitlab.com/victortoso/qapi-go/-/commits/qapi-golang-v3-by-tags
>
> --
>
> Sorry that its been awhile between v2 and v3, I had to prioritize other
> things. I hope to get this back on track in 2025.
>
> Cheers,
> Victor
>
> * Changes:
>
> On generated go:
>  - the output should be formatted as gofmt/goimports tools (Daniel)
>
>  - Included QAPI's documentation too (Daniel), see:
>    https://lists.gnu.org/archive/html/qemu-devel/2024-11/msg01621.html
>     
>  - Commands and Events should Marshal directly (Andrea)
>
> On python script:
>  - rebased: now uses QAPISchemaBranches, QAPISchemaAlternatives
>
>  - use textwrap as much as possible (Andrea)
>
>  - lots of changes to make the output like gofmt does
>
> Victor Toso (8):
>   qapi: golang: Generate enum type
>   qapi: golang: Generate alternate types
>   qapi: golang: Generate struct types
>   qapi: golang: structs: Address nullable members
>   qapi: golang: Generate union type
>   qapi: golang: Generate event type
>   qapi: golang: Generate command type
>   docs: add notes on Golang code generator
>
>  docs/devel/index-build.rst          |    1 +
>  docs/devel/qapi-golang-code-gen.rst |  548 +++++++++
>  scripts/qapi/golang.py              | 1645 +++++++++++++++++++++++++++
>  scripts/qapi/main.py                |    3 +
>  4 files changed, 2197 insertions(+)
>  create mode 100644 docs/devel/qapi-golang-code-gen.rst
>  create mode 100644 scripts/qapi/golang.py

This is series adds a backend that slots in cleanly, i.e. without any
changes to the core.  That makes it as low-risk to merge as it gets.

I'd like an Acked-by for the generated Go from someone familiar the kind
of software that could use it.

The Go backend is a single golang.py, which generates:

* Types:
  alternates.go enums.go structs.go unions.go

* Commands:
  commands.go

* Events:
  events.go

It doesn't generate visitors or introspection code.

Correct?

The existing C backend generates code for

* Types (types.py):
  qapi-builtin-types.[ch]
  qapi-types.[ch] qapi-types-*.[ch]

* Visitors (visit.py):

  qapi-builtin-visit.h
  qapi-visit.[ch] qapi-visit-*.[ch]

* Commands (commands.py):

  qapi-init-commands.h
  qapi-commands.[ch] qapi-commands-*.[ch]

* Events (events.py):

  qapi-emit-events.h
  qapi-events.[ch] qapi-events-*.[ch]

* Introspection (introspect.py):

  qapi-introspect.h

The -* files are all one pair of files per module (the things pulled in
with include directives), if any.  We do this to avoid "touch the QAPI
schema, recompile the world."

The generated Go is monolithic.  No "recompile the world" problem with
Go?

golang.py is somewhat big.  Whether splitting it up along the lines of
the C backend would improve things I can't say.  No need to worry about
that now.
Re: [PATCH v3 0/8] qapi-go: add generator for Golang interfaces
Posted by Daniel P. Berrangé 2 months, 2 weeks ago
On Mon, Jan 13, 2025 at 01:52:25PM +0100, Markus Armbruster wrote:
> Victor Toso <victortoso@redhat.com> writes:
> 
> > This patch series intent is to introduce a generator that produces a Go
> > module for Go applications to interact over QMP with QEMU.
> >
> > The initial Goal is to have a Go module that works as intended and can
> > be improved upon. I'd consider initial releases to be alpha while we
> > work with utilities tools and libraries on top of this.
> >
> > The generated code should reside in a separated Git repository, similar
> > to python-qemu-qmp.
> >
> > Applications should be able to consume this under qemu.org
> > namespace (e.g: import "qemu.org/go/qemu"), see Daniel's suggestion:
> > https://lists.gnu.org/archive/html/qemu-devel/2023-09/msg07024.html
> >
> > This is the third iteration:
> > v2: https://lists.gnu.org/archive/html/qemu-devel/2023-10/msg04785.html
> >
> > I've pushed this series in my gitlab fork:
> > https://gitlab.com/victortoso/qapi-go/
> >
> > The fork contains some tests, including tests that were generated from
> > QAPI's own examples from another generator created for testing, if you
> > are interested in it:
> > https://lists.gnu.org/archive/html/qemu-devel/2023-09/msg04946.html
> >
> > I've generated the qapi-go module over each commit of this series, see:
> > https://gitlab.com/victortoso/qapi-go/-/commits/qapi-golang-v3-by-patch
> >
> > I've also generated the qapi-go module over QEMU tags: v9.1.0, v9.2.0:
> > https://gitlab.com/victortoso/qapi-go/-/commits/qapi-golang-v3-by-tags
> >
> > --
> >
> > Sorry that its been awhile between v2 and v3, I had to prioritize other
> > things. I hope to get this back on track in 2025.
> >
> > Cheers,
> > Victor
> >
> > * Changes:
> >
> > On generated go:
> >  - the output should be formatted as gofmt/goimports tools (Daniel)
> >
> >  - Included QAPI's documentation too (Daniel), see:
> >    https://lists.gnu.org/archive/html/qemu-devel/2024-11/msg01621.html
> >     
> >  - Commands and Events should Marshal directly (Andrea)
> >
> > On python script:
> >  - rebased: now uses QAPISchemaBranches, QAPISchemaAlternatives
> >
> >  - use textwrap as much as possible (Andrea)
> >
> >  - lots of changes to make the output like gofmt does
> >
> > Victor Toso (8):
> >   qapi: golang: Generate enum type
> >   qapi: golang: Generate alternate types
> >   qapi: golang: Generate struct types
> >   qapi: golang: structs: Address nullable members
> >   qapi: golang: Generate union type
> >   qapi: golang: Generate event type
> >   qapi: golang: Generate command type
> >   docs: add notes on Golang code generator
> >
> >  docs/devel/index-build.rst          |    1 +
> >  docs/devel/qapi-golang-code-gen.rst |  548 +++++++++
> >  scripts/qapi/golang.py              | 1645 +++++++++++++++++++++++++++
> >  scripts/qapi/main.py                |    3 +
> >  4 files changed, 2197 insertions(+)
> >  create mode 100644 docs/devel/qapi-golang-code-gen.rst
> >  create mode 100644 scripts/qapi/golang.py
> 
> This is series adds a backend that slots in cleanly, i.e. without any
> changes to the core.  That makes it as low-risk to merge as it gets.
> 
> I'd like an Acked-by for the generated Go from someone familiar the kind
> of software that could use it.

In my other (huge) reply to this thread I tried to provide that
analysis by imagining how I would want to consume a QEMU API in
Go, creating an example application and trying to see how well
it fits with this design.

> The -* files are all one pair of files per module (the things pulled in
> with include directives), if any.  We do this to avoid "touch the QAPI
> schema, recompile the world."
> 
> The generated Go is monolithic.  No "recompile the world" problem with
> Go?

IIUC, the intent is that we dn't generate the go code as part of the
regular QEMU build. IOW, most of the time it would be generate once
against a release tag. If someone was actively working on QAPI schema
on git master, at the same time as developing a Go application, they'll
have a bit more of a repeated compile penalty. I'm not convinced that's
a big enough common case to worry about modularizing though.

> golang.py is somewhat big.  Whether splitting it up along the lines of
> the C backend would improve things I can't say.  No need to worry about
> that now.

With the monolothic go code, generated once per release tag, Go is
going to be reusing cached previously compiled objects most of the
time. I think that's likely to be good enough.

With regards,
Daniel
-- 
|: https://berrange.com      -o-    https://www.flickr.com/photos/dberrange :|
|: https://libvirt.org         -o-            https://fstop138.berrange.com :|
|: https://entangle-photo.org    -o-    https://www.instagram.com/dberrange :|
Re: [PATCH v3 0/8] qapi-go: add generator for Golang interfaces
Posted by Victor Toso 2 months, 3 weeks ago
Hi,

On Mon, Jan 13, 2025 at 01:52:25PM +0100, Markus Armbruster wrote:
> Victor Toso <victortoso@redhat.com> writes:
> 
> > This patch series intent is to introduce a generator that produces a Go
> > module for Go applications to interact over QMP with QEMU.
> >
> > The initial Goal is to have a Go module that works as intended and can
> > be improved upon. I'd consider initial releases to be alpha while we
> > work with utilities tools and libraries on top of this.
> >
> > The generated code should reside in a separated Git repository, similar
> > to python-qemu-qmp.
> >
> > Applications should be able to consume this under qemu.org
> > namespace (e.g: import "qemu.org/go/qemu"), see Daniel's suggestion:
> > https://lists.gnu.org/archive/html/qemu-devel/2023-09/msg07024.html
> >
> > This is the third iteration:
> > v2: https://lists.gnu.org/archive/html/qemu-devel/2023-10/msg04785.html
> >
> > I've pushed this series in my gitlab fork:
> > https://gitlab.com/victortoso/qapi-go/
> >
> > The fork contains some tests, including tests that were generated from
> > QAPI's own examples from another generator created for testing, if you
> > are interested in it:
> > https://lists.gnu.org/archive/html/qemu-devel/2023-09/msg04946.html
> >
> > I've generated the qapi-go module over each commit of this series, see:
> > https://gitlab.com/victortoso/qapi-go/-/commits/qapi-golang-v3-by-patch
> >
> > I've also generated the qapi-go module over QEMU tags: v9.1.0, v9.2.0:
> > https://gitlab.com/victortoso/qapi-go/-/commits/qapi-golang-v3-by-tags
> >
> > --
> >
> > Sorry that its been awhile between v2 and v3, I had to prioritize other
> > things. I hope to get this back on track in 2025.
> >
> > Cheers,
> > Victor
> >
> > * Changes:
> >
> > On generated go:
> >  - the output should be formatted as gofmt/goimports tools (Daniel)
> >
> >  - Included QAPI's documentation too (Daniel), see:
> >    https://lists.gnu.org/archive/html/qemu-devel/2024-11/msg01621.html
> >     
> >  - Commands and Events should Marshal directly (Andrea)
> >
> > On python script:
> >  - rebased: now uses QAPISchemaBranches, QAPISchemaAlternatives
> >
> >  - use textwrap as much as possible (Andrea)
> >
> >  - lots of changes to make the output like gofmt does
> >
> > Victor Toso (8):
> >   qapi: golang: Generate enum type
> >   qapi: golang: Generate alternate types
> >   qapi: golang: Generate struct types
> >   qapi: golang: structs: Address nullable members
> >   qapi: golang: Generate union type
> >   qapi: golang: Generate event type
> >   qapi: golang: Generate command type
> >   docs: add notes on Golang code generator
> >
> >  docs/devel/index-build.rst          |    1 +
> >  docs/devel/qapi-golang-code-gen.rst |  548 +++++++++
> >  scripts/qapi/golang.py              | 1645 +++++++++++++++++++++++++++
> >  scripts/qapi/main.py                |    3 +
> >  4 files changed, 2197 insertions(+)
> >  create mode 100644 docs/devel/qapi-golang-code-gen.rst
> >  create mode 100644 scripts/qapi/golang.py
> 
> This is series adds a backend that slots in cleanly, i.e. without any
> changes to the core.  That makes it as low-risk to merge as it gets.
> 
> I'd like an Acked-by for the generated Go from someone familiar the kind
> of software that could use it.
> 
> The Go backend is a single golang.py, which generates:
> 
> * Types:
>   alternates.go enums.go structs.go unions.go
> 
> * Commands:
>   commands.go
> 
> * Events:
>   events.go
> 
> It doesn't generate visitors or introspection code.
> 
> Correct?

Correct. I've actually thought about following what we did with
libvirt-go-module which appends _generated to the files that are
generated, meaning that we would also have files that are without
_generated in the name, with custoam/manual code related to that
file.

Either way, that does not change the Go module (does not break
API/ABI) so we can customize it if needed at a later time.
 
> The existing C backend generates code for
> 
> * Types (types.py):
>   qapi-builtin-types.[ch]
>   qapi-types.[ch] qapi-types-*.[ch]
> 
> * Visitors (visit.py):
> 
>   qapi-builtin-visit.h
>   qapi-visit.[ch] qapi-visit-*.[ch]
> 
> * Commands (commands.py):
> 
>   qapi-init-commands.h
>   qapi-commands.[ch] qapi-commands-*.[ch]
> 
> * Events (events.py):
> 
>   qapi-emit-events.h
>   qapi-events.[ch] qapi-events-*.[ch]
> 
> * Introspection (introspect.py):
> 
>   qapi-introspect.h
> 
> The -* files are all one pair of files per module (the things pulled in
> with include directives), if any.  We do this to avoid "touch the QAPI
> schema, recompile the world."
> 
> The generated Go is monolithic.  No "recompile the world" problem with
> Go?

From my experience, Go compiler only builds when the .go file
changes. If the QAPI changes and the generator outputs different
.go files, it'll recompile at the project using it.
 
> golang.py is somewhat big.  Whether splitting it up along the lines of
> the C backend would improve things I can't say.  No need to worry about
> that now.

Yes, I'm not super experienced with python. I'm all ears to any
suggestions on how to organize the source better. I'm happy to do
it as a follow-up.

Cheers,
Victor
Re: [PATCH v3 0/8] qapi-go: add generator for Golang interfaces
Posted by Daniel P. Berrangé 2 months, 2 weeks ago
On Fri, Jan 10, 2025 at 11:49:38AM +0100, Victor Toso wrote:
> I've pushed this series in my gitlab fork:
> https://gitlab.com/victortoso/qapi-go/
> 
> The fork contains some tests, including tests that were generated from
> QAPI's own examples from another generator created for testing, if you
> are interested in it:
> https://lists.gnu.org/archive/html/qemu-devel/2023-09/msg04946.html
> 
> I've generated the qapi-go module over each commit of this series, see:
> https://gitlab.com/victortoso/qapi-go/-/commits/qapi-golang-v3-by-patch
> 
> I've also generated the qapi-go module over QEMU tags: v9.1.0, v9.2.0:
> https://gitlab.com/victortoso/qapi-go/-/commits/qapi-golang-v3-by-tags

Apologies in advance for what will be a long mail.....

I can't recall all the discussions we had on previous versions now, but
I've been having a look at the generated code again, and have thoughts
around the design of the Event / Command types.


I've been wondering how to actually use the generated code to implement
a client and a server API. We previously said such APIs are out of scope
for this series, but at the same time I think they are pretty important
to consider as a way to evaluate the design approach.

Sending messages is easy as you always know what you're sending. The
big design issue is around receiving data off the wire, and the
associated unmarshalling. QMP has several different message types
(command, return, event, error, QMP greeting). When we've read a JSON
JSON doc off the wire, we (usually) have no idea what message type we
are going to unmarshal. Even if we know the message type, we then
might not know the sub-type.

IOW we have a major chicken & egg problem with unmarshalling.

Taking BalloonChangeEvent:

  type BalloonChangeEvent struct {
        MessageTimestamp Timestamp `json:"-"`
        // the logical size of the VM in bytes Formula used:
        // logical_vm_size = vm_ram_size - balloon_size
        Actual int64 `json:"actual"`
  }

which has an UnmarshallJSON method that consumes this whole doc:

    { "event": "BALLOON_CHANGE",
      "data": { "actual": 944766976 },
      "timestamp": { "seconds": 1267020223, "microseconds": 435656 } }

Before we can instantiate a BalloonChangeEvent struct and call its
UnmarshallJSON method, we have to already know we've received an
event and also know the event type is for BalloonChangeEvent.
This means we must have already partially unmarshalled it before
we can unmarshall into BalloonChangeEvent.

IMHO the idea of having UnmarshallJSON methods on the "Event" interface
(and "Command" interface) that consumes a complete message is thus
flawed.


Instead, I believe we need to have a two phase approach to unmarshalling.

For the 1st phase  we need a struct which can unmarshall *any* type of
QMP message, but only 1 level deep. Essentially something like this:

  type Message struct {
	QMP       *json.RawMessage `json:"QMP,omitempty"`
	Execute   string           `json:"execute,omitempty"`
	ExecOOB   string           `json:"exec-oob,omitempty"`
	Event     string           `json:"event,omitempty"`
	Error     *json.RawMessage `json:"error,omitempty"`
	Return    *json.RawMessage `json:"return,omitempty"`
	ID        string           `json:"id,omitempty"`
	Timestamp *Timestamp       `json:"timestamp,omitempty"`
	Data      *json.RawMessage `json:"data,omitempty"`
	Arguments *json.RawMessage `json:"arguments,omitempty"`
  }

Once that Message is unmarshalled, we can examine the 'QMP', 'Execute'
'ExecOOB', 'Event', 'Error' and 'Return' fields, to understand what type
of message we've received.

Then it is possible to identify the right Event or Command or Return
class to use to perform the second level of unmarshalling using the
json.RawMessage data the 1st level of unmarshalling captured.


IOW, if 'Event' is 'BALLOON_CHANGE_EVENT', we know we can use a
BalloonChangeEvent struct to Unmarshall. This struct, however, would
look different to what you've proposed, it would only need to contain
the 'data' fields for the event

    type BalloonChangeEvent struct {
        // the logical size of the VM in bytes Formula used:
        // logical_vm_size = vm_ram_size - balloon_size
        Actual int64 `json:"actual"`
    }

This would also mean there is no need to provide any UnmarshallJSON
or MarshallJSON methods at all on the BalloonChangeEvent, as the json
marshall code can work exclusively from the field annotations.

The same thoughts above apply to the command / return structs. I don't
think any of them need to have UnmarshallJSON/MarshallJSON methods.



    <<<< Pause here if you've read enough for now. >>>>



As a way to validate these thoughts, I spent a day to mock up a demo
of a QAPI client and server implementation.

First I created some manually written structs for the core QMP types

  https://gitlab.com/berrange/qapi-go-demo/-/blob/main/pkg/qapi/protocol.go

Then I've got (what would be) generated code creating types for the
specific QAPI schema. I've just illustrated this with query-machines
and qmp_capabilities commands, and STOP, CONT, BALLOON_CHANGE events:

  https://gitlab.com/berrange/qapi-go-demo/-/blob/main/pkg/qapi/gen_types.go

Essentially this is all the types you generated, but without any of
the (un)marshalling method impls which IMHO are all redundant. I put
everything in one file just for convenience, your per-file split makes
more sense in a real impl.


IMHO that's all that's needed to cover the scope of what is done in
this series, but to illustrate a real world usage model, I've then
gone on to implement both clients and servers.

At the very lowest level, both clients & servers need a way to send
and recv "Message" instances, so I created a "Channel" object with
SendMessage/RecvMessage methods:

  https://gitlab.com/berrange/qapi-go-demo/-/blob/main/pkg/qapi/channel.go

This is the most basic low level client one could have for QMP



From that we can derive a object for QMP Clients, giving slightly higher
level APIs to send commands, receive replies & events, avoiding the need
to touch the "Message" object directly:

  https://gitlab.com/berrange/qapi-go-demo/-/blob/main/pkg/qapi/client.go

And derive a similar object for QMP severs, giving APIs to dispatch commands,
and send replies, events & errors:

  https://gitlab.com/berrange/qapi-go-demo/-/blob/main/pkg/qapi/server.go

So far this is all static written code common to any QAPI schema impl.
The APIs can be used with any schema specific structs, however, the
latter might be defined.


Being low level, none of these APIs are strongly typed, which is not nice
as an application API in Go, so need a way to introduce better type safety.


In a real application I don't think developers should really be touching
the structs directly for commands/responses/events. Instead I believe
they need to be given formal APIs:

Thus I have (what would be) auto-generated interfaces for covering all
the commands and events:

  https://gitlab.com/berrange/qapi-go-demo/-/blob/main/pkg/qapi/gen_commands.go
  https://gitlab.com/berrange/qapi-go-demo/-/blob/main/pkg/qapi/gen_events.go

these interfaces are applicable to both clients and servers, just needing
different private implementations.

On the client side the implementations of these interfaces:

  https://gitlab.com/berrange/qapi-go-demo/-/blob/main/pkg/qapi/gen_client_sync.go
  https://gitlab.com/berrange/qapi-go-demo/-/blob/main/pkg/qapi/gen_client_async.go
  https://gitlab.com/berrange/qapi-go-demo/-/blob/main/pkg/qapi/gen_client_events.go

And on the server side the implementations of these interfaces

  https://gitlab.com/berrange/qapi-go-demo/-/blob/main/pkg/qapi/gen_server.go
  https://gitlab.com/berrange/qapi-go-demo/-/blob/main/pkg/qapi/gen_server_events.go


Finally we can consider what an implementation of a client looks like

  https://gitlab.com/berrange/qapi-go-demo/-/blob/main/cmd/sync-client/main.go
  https://gitlab.com/berrange/qapi-go-demo/-/blob/main/cmd/async-client/main.go

And what an implementation of a server looks like

  https://gitlab.com/berrange/qapi-go-demo/-/blob/main/cmd/server/main.go

The key observation at the end is that the client/server impls are all
using strongly typed APIs and don't have to concern themselves with JSON
at all.

The two demo clients should work with an real QEMU or the demo server.
The demo server won't work with qmp-shell, because I blindly assumed
each JSON message had a terminating newlinue for ease of impl and this
isn't compatible with qmp-shell. Solvable but I couldn't be bothered
for this demoware.


So what does this all mean wrt your series ? Not that much, probably
just a little code deletion and some small tweaks.


First, it illustrates that the approach taken for the Command / Event
interfaces & the corresponding UnmarshallJSON/MarshallJSON methods is
redundant code. Aside from that, I think everything else in your series
around generating the basic types is essentially good.

Second, it shows a way to build on top of your series, to define a higher
level API to make interacting with QMP much easier for app developers,
eliminating all exposure of JSON & marshalling, in favour of formal APIs.
This doesn't have to be done now, it can be a phase two, as long as we
have an approximate idea of what the phase two API will look like, so
we can validate the phase one design against these future plans.


NB What I've not especially considered in any of this is the impact of
differing QEMU versions & their schema changes. The easy way out is to
just re-run the generator for each version, putting them in a separate
Go package. So you can do 'import' of "qapi/qemu/abi920" for 9.2.0
schema APIs, or "qapi/qemu/910" for 9.1.0 schema APIs. The app dev would
have to choose which version (or versions) they consume & implement
against. Splitting versions across the whole package, avoids having to
consider versioning within parameters of a single command/event.

With regards,
Daniel
-- 
|: https://berrange.com      -o-    https://www.flickr.com/photos/dberrange :|
|: https://libvirt.org         -o-            https://fstop138.berrange.com :|
|: https://entangle-photo.org    -o-    https://www.instagram.com/dberrange :|
Re: [PATCH v3 0/8] qapi-go: add generator for Golang interfaces
Posted by Victor Toso 1 month, 3 weeks ago
Hi,

On Thu, Jan 16, 2025 at 09:59:52PM +0000, Daniel P. Berrangé wrote:
>     <<<< Pause here if you've read enough for now. >>>>
> 
> 
> 
> As a way to validate these thoughts, I spent a day to mock up a demo
> of a QAPI client and server implementation.
> 
> First I created some manually written structs for the core QMP types
> 
>   https://gitlab.com/berrange/qapi-go-demo/-/blob/main/pkg/qapi/protocol.go
> 
> Then I've got (what would be) generated code creating types for the
> specific QAPI schema. I've just illustrated this with query-machines
> and qmp_capabilities commands, and STOP, CONT, BALLOON_CHANGE events:
> 
>   https://gitlab.com/berrange/qapi-go-demo/-/blob/main/pkg/qapi/gen_types.go
> 
> Essentially this is all the types you generated, but without any of
> the (un)marshalling method impls which IMHO are all redundant. I put
> everything in one file just for convenience, your per-file split makes
> more sense in a real impl.
> 
> 
> IMHO that's all that's needed to cover the scope of what is done in
> this series, but to illustrate a real world usage model, I've then
> gone on to implement both clients and servers.
> 
> At the very lowest level, both clients & servers need a way to send
> and recv "Message" instances, so I created a "Channel" object with
> SendMessage/RecvMessage methods:
> 
>   https://gitlab.com/berrange/qapi-go-demo/-/blob/main/pkg/qapi/channel.go
> 
> This is the most basic low level client one could have for QMP
> 
> 
> 
> From that we can derive a object for QMP Clients, giving slightly higher
> level APIs to send commands, receive replies & events, avoiding the need
> to touch the "Message" object directly:
> 
>   https://gitlab.com/berrange/qapi-go-demo/-/blob/main/pkg/qapi/client.go
> 
> And derive a similar object for QMP severs, giving APIs to dispatch commands,
> and send replies, events & errors:
> 
>   https://gitlab.com/berrange/qapi-go-demo/-/blob/main/pkg/qapi/server.go
> 
> So far this is all static written code common to any QAPI schema impl.
> The APIs can be used with any schema specific structs, however, the
> latter might be defined.
> 
> 
> Being low level, none of these APIs are strongly typed, which is not nice
> as an application API in Go, so need a way to introduce better type safety.
> 
> 
> In a real application I don't think developers should really be touching
> the structs directly for commands/responses/events. Instead I believe
> they need to be given formal APIs:
> 
> Thus I have (what would be) auto-generated interfaces for covering all
> the commands and events:
> 
>   https://gitlab.com/berrange/qapi-go-demo/-/blob/main/pkg/qapi/gen_commands.go
>   https://gitlab.com/berrange/qapi-go-demo/-/blob/main/pkg/qapi/gen_events.go

One thing that bothers me a little is using event's fields
directly instead of event type in the interface itself:

    type Events interface {
        StopEvent(time.Time) error
        ContEvent(time.Time) error
        BalloonChangeEvent(int64, time.Time) error
    }

In the gen_server_events.go:

    func (cmds *serverEvents) BalloonChangeEvent(actual int64, when time.Time) error {
        ev := &BalloonChangeEvent{
            Actual: actual,
        }
        return cmds.QMP.SendEvent("BALLOON_CHANGE", ev, when)
    }

I'm actually in favor of moving the Type itself to the method:

    type Events interface {
        StopEvent(time.Time) error
        ContEvent(time.Time) error
        BalloonChangeEvent(BalloonChangeEvent, time.Time) error
    }

    func (cmds *serverEvents) BalloonChangeEvent(ev BalloonChangeEvent, when time.Time) error {
        return cmds.QMP.SendEvent("BALLOON_CHANGE", &ev, when)
    }


For the caller it might be an extra step but I see a few
benefits:

    1. Documentation: I'm generating the QAPI documentation of
       the events in its types. IDE's can jump-to-definition and
       see the documentation there. Obviously we could move the
       documentation here or duplicate it but the doc in the Type
       looks the right place imho.
    
    2. Changes in the Event fields over time would not impact the
       Breaking the Methods API. While some fields my be
       added/removed/changed, I think this would help break less
       the application when bumping the go module version.


I'm not sure about empty types like StopEvent. I'm generating it
with its doc but as it goes, it is bound to not be used, being
generated just as source of documentation. Still, if it so
happens that it does change in the future, we would need to
extend the Method with its field or its type.

    // Emitted when the virtual machine is stopped
    //
    // Since: 0.12
    //
    // .. qmp-example::    <- { "event": "STOP",      "timestamp": {
    // "seconds": 1267041730, "microseconds": 281295 } }
    type StopEvent struct{}


That also applies for commands..

    // Resume guest VM execution.
    //
    // Since: 0.14
    //
    // .. note:: This command will succeed if the guest is currently
    // running. It will also succeed if the guest is in the "inmigrate"
    // state; in this case, the effect of the command is to make sure
    // the guest starts once migration finishes, removing the effect of
    // the "-S" command line option if it was passed.    If the VM was
    // previously suspended, and not been reset or woken,   this command
    // will transition back to the "suspended" state.   (Since 9.0)  ..
    // qmp-example::    -> { "execute": "cont" }   <- { "return": {} }
    type ContCommand struct {}


I honestly did the requested changes a little while ago but I was
thinking about this and tinkering a little with the demo.

Cheers,
Victor

> these interfaces are applicable to both clients and servers, just needing
> different private implementations.
> 
> On the client side the implementations of these interfaces:
> 
>   https://gitlab.com/berrange/qapi-go-demo/-/blob/main/pkg/qapi/gen_client_sync.go
>   https://gitlab.com/berrange/qapi-go-demo/-/blob/main/pkg/qapi/gen_client_async.go
>   https://gitlab.com/berrange/qapi-go-demo/-/blob/main/pkg/qapi/gen_client_events.go
> 
> And on the server side the implementations of these interfaces
> 
>   https://gitlab.com/berrange/qapi-go-demo/-/blob/main/pkg/qapi/gen_server.go
>   https://gitlab.com/berrange/qapi-go-demo/-/blob/main/pkg/qapi/gen_server_events.go
> 
> 
> Finally we can consider what an implementation of a client looks like
> 
>   https://gitlab.com/berrange/qapi-go-demo/-/blob/main/cmd/sync-client/main.go
>   https://gitlab.com/berrange/qapi-go-demo/-/blob/main/cmd/async-client/main.go
> 
> And what an implementation of a server looks like
> 
>   https://gitlab.com/berrange/qapi-go-demo/-/blob/main/cmd/server/main.go
> 
> The key observation at the end is that the client/server impls are all
> using strongly typed APIs and don't have to concern themselves with JSON
> at all.
> 
> The two demo clients should work with an real QEMU or the demo server.
> The demo server won't work with qmp-shell, because I blindly assumed
> each JSON message had a terminating newlinue for ease of impl and this
> isn't compatible with qmp-shell. Solvable but I couldn't be bothered
> for this demoware.
> 
> 
> So what does this all mean wrt your series ? Not that much, probably
> just a little code deletion and some small tweaks.
> 
> 
> First, it illustrates that the approach taken for the Command / Event
> interfaces & the corresponding UnmarshallJSON/MarshallJSON methods is
> redundant code. Aside from that, I think everything else in your series
> around generating the basic types is essentially good.
> 
> Second, it shows a way to build on top of your series, to define a higher
> level API to make interacting with QMP much easier for app developers,
> eliminating all exposure of JSON & marshalling, in favour of formal APIs.
> This doesn't have to be done now, it can be a phase two, as long as we
> have an approximate idea of what the phase two API will look like, so
> we can validate the phase one design against these future plans.
> 
> 
> NB What I've not especially considered in any of this is the impact of
> differing QEMU versions & their schema changes. The easy way out is to
> just re-run the generator for each version, putting them in a separate
> Go package. So you can do 'import' of "qapi/qemu/abi920" for 9.2.0
> schema APIs, or "qapi/qemu/910" for 9.1.0 schema APIs. The app dev would
> have to choose which version (or versions) they consume & implement
> against. Splitting versions across the whole package, avoids having to
> consider versioning within parameters of a single command/event.
> 
> With regards,
> Daniel
> -- 
> |: https://berrange.com      -o-    https://www.flickr.com/photos/dberrange :|
> |: https://libvirt.org         -o-            https://fstop138.berrange.com :|
> |: https://entangle-photo.org    -o-    https://www.instagram.com/dberrange :|
> 
Re: [PATCH v3 0/8] qapi-go: add generator for Golang interfaces
Posted by Daniel P. Berrangé 1 month, 3 weeks ago
On Tue, Feb 11, 2025 at 11:25:05AM +0100, Victor Toso wrote:
> Hi,
> 
> On Thu, Jan 16, 2025 at 09:59:52PM +0000, Daniel P. Berrangé wrote:
> >     <<<< Pause here if you've read enough for now. >>>>
> > 
> > 
> > 
> > As a way to validate these thoughts, I spent a day to mock up a demo
> > of a QAPI client and server implementation.
> > 
> > First I created some manually written structs for the core QMP types
> > 
> >   https://gitlab.com/berrange/qapi-go-demo/-/blob/main/pkg/qapi/protocol.go
> > 
> > Then I've got (what would be) generated code creating types for the
> > specific QAPI schema. I've just illustrated this with query-machines
> > and qmp_capabilities commands, and STOP, CONT, BALLOON_CHANGE events:
> > 
> >   https://gitlab.com/berrange/qapi-go-demo/-/blob/main/pkg/qapi/gen_types.go
> > 
> > Essentially this is all the types you generated, but without any of
> > the (un)marshalling method impls which IMHO are all redundant. I put
> > everything in one file just for convenience, your per-file split makes
> > more sense in a real impl.
> > 
> > 
> > IMHO that's all that's needed to cover the scope of what is done in
> > this series, but to illustrate a real world usage model, I've then
> > gone on to implement both clients and servers.
> > 
> > At the very lowest level, both clients & servers need a way to send
> > and recv "Message" instances, so I created a "Channel" object with
> > SendMessage/RecvMessage methods:
> > 
> >   https://gitlab.com/berrange/qapi-go-demo/-/blob/main/pkg/qapi/channel.go
> > 
> > This is the most basic low level client one could have for QMP
> > 
> > 
> > 
> > From that we can derive a object for QMP Clients, giving slightly higher
> > level APIs to send commands, receive replies & events, avoiding the need
> > to touch the "Message" object directly:
> > 
> >   https://gitlab.com/berrange/qapi-go-demo/-/blob/main/pkg/qapi/client.go
> > 
> > And derive a similar object for QMP severs, giving APIs to dispatch commands,
> > and send replies, events & errors:
> > 
> >   https://gitlab.com/berrange/qapi-go-demo/-/blob/main/pkg/qapi/server.go
> > 
> > So far this is all static written code common to any QAPI schema impl.
> > The APIs can be used with any schema specific structs, however, the
> > latter might be defined.
> > 
> > 
> > Being low level, none of these APIs are strongly typed, which is not nice
> > as an application API in Go, so need a way to introduce better type safety.
> > 
> > 
> > In a real application I don't think developers should really be touching
> > the structs directly for commands/responses/events. Instead I believe
> > they need to be given formal APIs:
> > 
> > Thus I have (what would be) auto-generated interfaces for covering all
> > the commands and events:
> > 
> >   https://gitlab.com/berrange/qapi-go-demo/-/blob/main/pkg/qapi/gen_commands.go
> >   https://gitlab.com/berrange/qapi-go-demo/-/blob/main/pkg/qapi/gen_events.go
> 
> One thing that bothers me a little is using event's fields
> directly instead of event type in the interface itself:
> 
>     type Events interface {
>         StopEvent(time.Time) error
>         ContEvent(time.Time) error
>         BalloonChangeEvent(int64, time.Time) error
>     }
> 
> In the gen_server_events.go:
> 
>     func (cmds *serverEvents) BalloonChangeEvent(actual int64, when time.Time) error {
>         ev := &BalloonChangeEvent{
>             Actual: actual,
>         }
>         return cmds.QMP.SendEvent("BALLOON_CHANGE", ev, when)
>     }
> 
> I'm actually in favor of moving the Type itself to the method:
> 
>     type Events interface {
>         StopEvent(time.Time) error
>         ContEvent(time.Time) error
>         BalloonChangeEvent(BalloonChangeEvent, time.Time) error
>     }
> 
>     func (cmds *serverEvents) BalloonChangeEvent(ev BalloonChangeEvent, when time.Time) error {
>         return cmds.QMP.SendEvent("BALLOON_CHANGE", &ev, when)
>     }

I choose to expand all the types inline in events & commands
because I felt like this gives a more convenient API design
for applications, without the extra indirectino of packing
and unpacking parameters
.
> For the caller it might be an extra step but I see a few
> benefits:
> 
>     1. Documentation: I'm generating the QAPI documentation of
>        the events in its types. IDE's can jump-to-definition and
>        see the documentation there. Obviously we could move the
>        documentation here or duplicate it but the doc in the Type
>        looks the right place imho.

Yep, it would complicate docs to have to unpack it against the
method. I was coming at this from the POV of an application
developer though, intentionally ignoring what is the most
convenient for the code generator. The latter is fixed cost
while writing the generator, while the API design is an ongoing
cost time, so makes more sense to optimize for the latter.

>     2. Changes in the Event fields over time would not impact the
>        Breaking the Methods API. While some fields my be
>        added/removed/changed, I think this would help break less
>        the application when bumping the go module version.

Yep, hiding everything behind a struct can reduce breakage.

The tricky (impossible) question is how beneficial it is in
the real world usage ?

As guidance, taking protobuf as a commonly used package though,
if I look at Kubevirt's protobuf generated APIs:

  https://github.com/kubevirt/kubevirt/blob/main/pkg/handler-launcher-com/cmd/v1/cmd.pb.go#L1237

I can see they're using structs for parameters and return
values. So that widely used prior art, suggests we go the
way you outline and use structs.


> I'm not sure about empty types like StopEvent. I'm generating it
> with its doc but as it goes, it is bound to not be used, being
> generated just as source of documentation. Still, if it so
> happens that it does change in the future, we would need to
> extend the Method with its field or its type.
> 
>     // Emitted when the virtual machine is stopped
>     //
>     // Since: 0.12
>     //
>     // .. qmp-example::    <- { "event": "STOP",      "timestamp": {
>     // "seconds": 1267041730, "microseconds": 281295 } }
>     type StopEvent struct{}

Yes, if the goal of using structs is to reduce breakage when
fields are added/removed, then IMHO the logical conclusion
is that empty structs must be generated.

> That also applies for commands..
> 
>     // Resume guest VM execution.
>     //
>     // Since: 0.14
>     //
>     // .. note:: This command will succeed if the guest is currently
>     // running. It will also succeed if the guest is in the "inmigrate"
>     // state; in this case, the effect of the command is to make sure
>     // the guest starts once migration finishes, removing the effect of
>     // the "-S" command line option if it was passed.    If the VM was
>     // previously suspended, and not been reset or woken,   this command
>     // will transition back to the "suspended" state.   (Since 9.0)  ..
>     // qmp-example::    -> { "execute": "cont" }   <- { "return": {} }
>     type ContCommand struct {}
> 
> 
> I honestly did the requested changes a little while ago but I was
> thinking about this and tinkering a little with the demo.

With regards,
Daniel
-- 
|: https://berrange.com      -o-    https://www.flickr.com/photos/dberrange :|
|: https://libvirt.org         -o-            https://fstop138.berrange.com :|
|: https://entangle-photo.org    -o-    https://www.instagram.com/dberrange :|


Re: [PATCH v3 0/8] qapi-go: add generator for Golang interfaces
Posted by Victor Toso 1 month, 3 weeks ago
Hi,

On Tue, Feb 11, 2025 at 11:10:37AM +0000, Daniel P. Berrangé wrote:
> On Tue, Feb 11, 2025 at 11:25:05AM +0100, Victor Toso wrote:
> > Hi,
> > 
> > On Thu, Jan 16, 2025 at 09:59:52PM +0000, Daniel P. Berrangé wrote:
> > >     <<<< Pause here if you've read enough for now. >>>>
> > > 
> > > 
> > > 
> > > As a way to validate these thoughts, I spent a day to mock up a demo
> > > of a QAPI client and server implementation.
> > > 
> > > First I created some manually written structs for the core QMP types
> > > 
> > >   https://gitlab.com/berrange/qapi-go-demo/-/blob/main/pkg/qapi/protocol.go
> > > 
> > > Then I've got (what would be) generated code creating types for the
> > > specific QAPI schema. I've just illustrated this with query-machines
> > > and qmp_capabilities commands, and STOP, CONT, BALLOON_CHANGE events:
> > > 
> > >   https://gitlab.com/berrange/qapi-go-demo/-/blob/main/pkg/qapi/gen_types.go
> > > 
> > > Essentially this is all the types you generated, but without any of
> > > the (un)marshalling method impls which IMHO are all redundant. I put
> > > everything in one file just for convenience, your per-file split makes
> > > more sense in a real impl.
> > > 
> > > 
> > > IMHO that's all that's needed to cover the scope of what is done in
> > > this series, but to illustrate a real world usage model, I've then
> > > gone on to implement both clients and servers.
> > > 
> > > At the very lowest level, both clients & servers need a way to send
> > > and recv "Message" instances, so I created a "Channel" object with
> > > SendMessage/RecvMessage methods:
> > > 
> > >   https://gitlab.com/berrange/qapi-go-demo/-/blob/main/pkg/qapi/channel.go
> > > 
> > > This is the most basic low level client one could have for QMP
> > > 
> > > 
> > > 
> > > From that we can derive a object for QMP Clients, giving slightly higher
> > > level APIs to send commands, receive replies & events, avoiding the need
> > > to touch the "Message" object directly:
> > > 
> > >   https://gitlab.com/berrange/qapi-go-demo/-/blob/main/pkg/qapi/client.go
> > > 
> > > And derive a similar object for QMP severs, giving APIs to dispatch commands,
> > > and send replies, events & errors:
> > > 
> > >   https://gitlab.com/berrange/qapi-go-demo/-/blob/main/pkg/qapi/server.go
> > > 
> > > So far this is all static written code common to any QAPI schema impl.
> > > The APIs can be used with any schema specific structs, however, the
> > > latter might be defined.
> > > 
> > > 
> > > Being low level, none of these APIs are strongly typed, which is not nice
> > > as an application API in Go, so need a way to introduce better type safety.
> > > 
> > > 
> > > In a real application I don't think developers should really be touching
> > > the structs directly for commands/responses/events. Instead I believe
> > > they need to be given formal APIs:
> > > 
> > > Thus I have (what would be) auto-generated interfaces for covering all
> > > the commands and events:
> > > 
> > >   https://gitlab.com/berrange/qapi-go-demo/-/blob/main/pkg/qapi/gen_commands.go
> > >   https://gitlab.com/berrange/qapi-go-demo/-/blob/main/pkg/qapi/gen_events.go
> > 
> > One thing that bothers me a little is using event's fields
> > directly instead of event type in the interface itself:
> > 
> >     type Events interface {
> >         StopEvent(time.Time) error
> >         ContEvent(time.Time) error
> >         BalloonChangeEvent(int64, time.Time) error
> >     }
> > 
> > In the gen_server_events.go:
> > 
> >     func (cmds *serverEvents) BalloonChangeEvent(actual int64, when time.Time) error {
> >         ev := &BalloonChangeEvent{
> >             Actual: actual,
> >         }
> >         return cmds.QMP.SendEvent("BALLOON_CHANGE", ev, when)
> >     }
> > 
> > I'm actually in favor of moving the Type itself to the method:
> > 
> >     type Events interface {
> >         StopEvent(time.Time) error
> >         ContEvent(time.Time) error
> >         BalloonChangeEvent(BalloonChangeEvent, time.Time) error
> >     }
> > 
> >     func (cmds *serverEvents) BalloonChangeEvent(ev BalloonChangeEvent, when time.Time) error {
> >         return cmds.QMP.SendEvent("BALLOON_CHANGE", &ev, when)
> >     }
> 
> I choose to expand all the types inline in events & commands
> because I felt like this gives a more convenient API design
> for applications, without the extra indirectino of packing
> and unpacking parameters

Sure, just a choice to be made.

> > For the caller it might be an extra step but I see a few
> > benefits:
> > 
> >     1. Documentation: I'm generating the QAPI documentation of
> >        the events in its types. IDE's can jump-to-definition and
> >        see the documentation there. Obviously we could move the
> >        documentation here or duplicate it but the doc in the Type
> >        looks the right place imho.
> 
> Yep, it would complicate docs to have to unpack it against the
> method. I was coming at this from the POV of an application
> developer though, intentionally ignoring what is the most
> convenient for the code generator. The latter is fixed cost
> while writing the generator, while the API design is an ongoing
> cost time, so makes more sense to optimize for the latter.

Right. I don't mind doing the work to inline it, but I'm not sure
it is best. Inline commands/events that don't have fields are
straight forward and look nice but what when they have several
fields instead?

Wrt to mandatory and optional fields, I think it is
straightforward when looking at the struct definition that might
be confusing to the methods args.

As example, block-stream command has 12 fields, only 1 mandatory.

    // .. qmp-example::    -> { "execute": "block-stream",
    // "arguments": { "device": "virtio0",             "base":
    // "/tmp/master.qcow2" } }   <- { "return": {} }
    type BlockStreamCommand struct {
        // identifier for the newly-created block job. If omitted, the
        // device name will be used. (Since 2.7)
        JobId *string `json:"job-id,omitempty"`
        // the device or node name of the top image
        Device string `json:"device"`
        // the common backing file name. It cannot be set if @base-node or
        // @bottom is also set.
        Base *string `json:"base,omitempty"`
        // the node name of the backing file. It cannot be set if @base or
        // @bottom is also set. (Since 2.8)
        BaseNode *string `json:"base-node,omitempty"`
        // The backing file string to write into the top image. This
        // filename is not validated. If a pathname string is such that it
        // cannot be resolved by QEMU, that means that subsequent QMP or
        // HMP commands must use node-names for the image in question, as
        // filename lookup methods will fail. If not specified, QEMU will
        // automatically determine the backing file string to use, or
        // error out if there is no obvious choice. Care should be taken
        // when specifying the string, to specify a valid filename or
        // protocol. (Since 2.1)
        BackingFile *string `json:"backing-file,omitempty"`
        // If true, replace any protocol mentioned in the 'backing file
        // format' with 'raw', rather than storing the protocol name as
        // the backing format. Can be used even when no image header will
        // be updated (default false; since 9.0).
        BackingMaskProtocol *bool `json:"backing-mask-protocol,omitempty"`
        // the last node in the chain that should be streamed into top. It
        // cannot be set if @base or @base-node is also set. It cannot be
        // filter node. (Since 6.0)
        Bottom *string `json:"bottom,omitempty"`
        // the maximum speed, in bytes per second
        Speed *int64 `json:"speed,omitempty"`
        // the action to take on an error (default report). 'stop' and
        // 'enospc' can only be used if the block device supports io-
        // status (see BlockInfo). (Since 1.3)
        OnError *BlockdevOnError `json:"on-error,omitempty"`
        // the node name that should be assigned to the filter driver that
        // the stream job inserts into the graph above @device. If this
        // option is not given, a node name is autogenerated. (Since: 6.0)
        FilterNodeName *string `json:"filter-node-name,omitempty"`
        // When false, this job will wait in a PENDING state after it has
        // finished its work, waiting for @block-job-finalize before
        // making any block graph changes. When true, this job will
        // automatically perform its abort or commit actions. Defaults to
        // true. (Since 3.1)
        AutoFinalize *bool `json:"auto-finalize,omitempty"`
        // When false, this job will wait in a CONCLUDED state after it
        // has completely ceased all work, and awaits @block-job-dismiss.
        // When true, this job will automatically disappear from the query
        // list without user intervention. Defaults to true. (Since 3.1)
        AutoDismiss *bool `json:"auto-dismiss,omitempty"`
    }

Might be easier in that case to

    cmds.BlockStream(BlockStreamCommand{device:"virtio"})

instead of defining all optional args as nil.

 
> >     2. Changes in the Event fields over time would not impact the
> >        Breaking the Methods API. While some fields my be
> >        added/removed/changed, I think this would help break less
> >        the application when bumping the go module version.
> 
> Yep, hiding everything behind a struct can reduce breakage.
> 
> The tricky (impossible) question is how beneficial it is in
> the real world usage ?

To me it was just an extra benefit, not a vital one. We are bound
to break when user bumps it. In the real world, as far as I
recall from discussing this with Markus (feel free to correct me)

    i.  Removing fields is somewhat rare. I remember he put out
        an example but I could not find it.
    ii. Adding fields is not so rare although they are usually
        optional.
 
With inlining approach we would break application in both cases
100% of the time. It'd be less so by using a type.

> As guidance, taking protobuf as a commonly used package though,
> if I look at Kubevirt's protobuf generated APIs:
> 
>   https://github.com/kubevirt/kubevirt/blob/main/pkg/handler-launcher-com/cmd/v1/cmd.pb.go#L1237
> 
> I can see they're using structs for parameters and return
> values. So that widely used prior art, suggests we go the
> way you outline and use structs.

Oh, but if I'm not mistaken, if you do change the type you are
required to bump the version meaning that you'll need to
generated a new set of types with the new version. That enforces
client/side to implement the new type if they want to talk over
new version (enforced by capabilites negotiation). Not a great
dev experience but it surely works and performance is good too...
but I digress.
 
> > I'm not sure about empty types like StopEvent. I'm generating it
> > with its doc but as it goes, it is bound to not be used, being
> > generated just as source of documentation. Still, if it so
> > happens that it does change in the future, we would need to
> > extend the Method with its field or its type.
> > 
> >     // Emitted when the virtual machine is stopped
> >     //
> >     // Since: 0.12
> >     //
> >     // .. qmp-example::    <- { "event": "STOP",      "timestamp": {
> >     // "seconds": 1267041730, "microseconds": 281295 } }
> >     type StopEvent struct{}
> 
> Yes, if the goal of using structs is to reduce breakage when
> fields are added/removed, then IMHO the logical conclusion
> is that empty structs must be generated.

imho, just a extra benefit. Breaking is somewhat fine if it
happens only when we bump the version and is not something that
happens so often that it becomes a bother.
 
> > That also applies for commands..
> > 
> >     // Resume guest VM execution.
> >     //
> >     // Since: 0.14
> >     //
> >     // .. note:: This command will succeed if the guest is currently
> >     // running. It will also succeed if the guest is in the "inmigrate"
> >     // state; in this case, the effect of the command is to make sure
> >     // the guest starts once migration finishes, removing the effect of
> >     // the "-S" command line option if it was passed.    If the VM was
> >     // previously suspended, and not been reset or woken,   this command
> >     // will transition back to the "suspended" state.   (Since 9.0)  ..
> >     // qmp-example::    -> { "execute": "cont" }   <- { "return": {} }
> >     type ContCommand struct {}
> > 
> > 
> > I honestly did the requested changes a little while ago but I was
> > thinking about this and tinkering a little with the demo.
> 
> With regards,
> Daniel

Thanks for the quick replies,
Victor
Re: [PATCH v3 0/8] qapi-go: add generator for Golang interfaces
Posted by Victor Toso 2 months ago
Hi,

On Thu, Jan 16, 2025 at 09:59:52PM +0000, Daniel P. Berrangé wrote:
> On Fri, Jan 10, 2025 at 11:49:38AM +0100, Victor Toso wrote:
> > I've pushed this series in my gitlab fork:
> > https://gitlab.com/victortoso/qapi-go/
> > 
> > The fork contains some tests, including tests that were generated from
> > QAPI's own examples from another generator created for testing, if you
> > are interested in it:
> > https://lists.gnu.org/archive/html/qemu-devel/2023-09/msg04946.html
> > 
> > I've generated the qapi-go module over each commit of this series, see:
> > https://gitlab.com/victortoso/qapi-go/-/commits/qapi-golang-v3-by-patch
> > 
> > I've also generated the qapi-go module over QEMU tags: v9.1.0, v9.2.0:
> > https://gitlab.com/victortoso/qapi-go/-/commits/qapi-golang-v3-by-tags
> 
> Apologies in advance for what will be a long mail.....
> 
> I can't recall all the discussions we had on previous versions now, but
> I've been having a look at the generated code again, and have thoughts
> around the design of the Event / Command types.
> 
> 
> I've been wondering how to actually use the generated code to implement
> a client and a server API. We previously said such APIs are out of scope
> for this series, but at the same time I think they are pretty important
> to consider as a way to evaluate the design approach.
> 
> Sending messages is easy as you always know what you're sending. The
> big design issue is around receiving data off the wire, and the
> associated unmarshalling. QMP has several different message types
> (command, return, event, error, QMP greeting). When we've read a JSON
> JSON doc off the wire, we (usually) have no idea what message type we
> are going to unmarshal. Even if we know the message type, we then
> might not know the sub-type.
> 
> IOW we have a major chicken & egg problem with unmarshalling.
> 
> Taking BalloonChangeEvent:
> 
>   type BalloonChangeEvent struct {
>         MessageTimestamp Timestamp `json:"-"`
>         // the logical size of the VM in bytes Formula used:
>         // logical_vm_size = vm_ram_size - balloon_size
>         Actual int64 `json:"actual"`
>   }
> 
> which has an UnmarshallJSON method that consumes this whole doc:
> 
>     { "event": "BALLOON_CHANGE",
>       "data": { "actual": 944766976 },
>       "timestamp": { "seconds": 1267020223, "microseconds": 435656 } }
> 
> Before we can instantiate a BalloonChangeEvent struct and call its
> UnmarshallJSON method, we have to already know we've received an
> event and also know the event type is for BalloonChangeEvent.
> This means we must have already partially unmarshalled it before
> we can unmarshall into BalloonChangeEvent.
> 
> IMHO the idea of having UnmarshallJSON methods on the "Event" interface
> (and "Command" interface) that consumes a complete message is thus
> flawed.
> 
> 
> Instead, I believe we need to have a two phase approach to unmarshalling.
> 
> For the 1st phase  we need a struct which can unmarshall *any* type of
> QMP message, but only 1 level deep. Essentially something like this:
> 
>   type Message struct {
> 	QMP       *json.RawMessage `json:"QMP,omitempty"`
> 	Execute   string           `json:"execute,omitempty"`
> 	ExecOOB   string           `json:"exec-oob,omitempty"`
> 	Event     string           `json:"event,omitempty"`
> 	Error     *json.RawMessage `json:"error,omitempty"`
> 	Return    *json.RawMessage `json:"return,omitempty"`
> 	ID        string           `json:"id,omitempty"`
> 	Timestamp *Timestamp       `json:"timestamp,omitempty"`
> 	Data      *json.RawMessage `json:"data,omitempty"`
> 	Arguments *json.RawMessage `json:"arguments,omitempty"`
>   }

/* Note to self, should have probably tested 'exec-oob' */
 
> Once that Message is unmarshalled, we can examine the 'QMP', 'Execute'
> 'ExecOOB', 'Event', 'Error' and 'Return' fields, to understand what type
> of message we've received.
> 
> Then it is possible to identify the right Event or Command or Return
> class to use to perform the second level of unmarshalling using the
> json.RawMessage data the 1st level of unmarshalling captured.
> 
> 
> IOW, if 'Event' is 'BALLOON_CHANGE_EVENT', we know we can use a
> BalloonChangeEvent struct to Unmarshall. This struct, however, would
> look different to what you've proposed, it would only need to contain
> the 'data' fields for the event
> 
>     type BalloonChangeEvent struct {
>         // the logical size of the VM in bytes Formula used:
>         // logical_vm_size = vm_ram_size - balloon_size
>         Actual int64 `json:"actual"`
>     }
> 
> This would also mean there is no need to provide any UnmarshallJSON
> or MarshallJSON methods at all on the BalloonChangeEvent, as the json
> marshall code can work exclusively from the field annotations.
> 
> The same thoughts above apply to the command / return structs. I don't
> think any of them need to have UnmarshallJSON/MarshallJSON methods.
> 
> 
> 
>     <<<< Pause here if you've read enough for now. >>>>
> 
> 
> 
> As a way to validate these thoughts, I spent a day to mock up a demo
> of a QAPI client and server implementation.
> 
> First I created some manually written structs for the core QMP types
> 
>   https://gitlab.com/berrange/qapi-go-demo/-/blob/main/pkg/qapi/protocol.go
> 
> Then I've got (what would be) generated code creating types for the
> specific QAPI schema. I've just illustrated this with query-machines
> and qmp_capabilities commands, and STOP, CONT, BALLOON_CHANGE events:
> 
>   https://gitlab.com/berrange/qapi-go-demo/-/blob/main/pkg/qapi/gen_types.go
> 
> Essentially this is all the types you generated, but without any of
> the (un)marshalling method impls which IMHO are all redundant. I put
> everything in one file just for convenience, your per-file split makes
> more sense in a real impl.
> 
> 
> IMHO that's all that's needed to cover the scope of what is done in
> this series, but to illustrate a real world usage model, I've then
> gone on to implement both clients and servers.
> 
> At the very lowest level, both clients & servers need a way to send
> and recv "Message" instances, so I created a "Channel" object with
> SendMessage/RecvMessage methods:
> 
>   https://gitlab.com/berrange/qapi-go-demo/-/blob/main/pkg/qapi/channel.go
> 
> This is the most basic low level client one could have for QMP
> 
> 
> 
> From that we can derive a object for QMP Clients, giving slightly higher
> level APIs to send commands, receive replies & events, avoiding the need
> to touch the "Message" object directly:
> 
>   https://gitlab.com/berrange/qapi-go-demo/-/blob/main/pkg/qapi/client.go
> 
> And derive a similar object for QMP severs, giving APIs to dispatch commands,
> and send replies, events & errors:
> 
>   https://gitlab.com/berrange/qapi-go-demo/-/blob/main/pkg/qapi/server.go
> 
> So far this is all static written code common to any QAPI schema impl.
> The APIs can be used with any schema specific structs, however, the
> latter might be defined.
> 
> 
> Being low level, none of these APIs are strongly typed, which is not nice
> as an application API in Go, so need a way to introduce better type safety.
> 
> 
> In a real application I don't think developers should really be touching
> the structs directly for commands/responses/events. Instead I believe
> they need to be given formal APIs:
> 
> Thus I have (what would be) auto-generated interfaces for covering all
> the commands and events:
> 
>   https://gitlab.com/berrange/qapi-go-demo/-/blob/main/pkg/qapi/gen_commands.go
>   https://gitlab.com/berrange/qapi-go-demo/-/blob/main/pkg/qapi/gen_events.go
> 
> these interfaces are applicable to both clients and servers, just needing
> different private implementations.
> 
> On the client side the implementations of these interfaces:
> 
>   https://gitlab.com/berrange/qapi-go-demo/-/blob/main/pkg/qapi/gen_client_sync.go
>   https://gitlab.com/berrange/qapi-go-demo/-/blob/main/pkg/qapi/gen_client_async.go
>   https://gitlab.com/berrange/qapi-go-demo/-/blob/main/pkg/qapi/gen_client_events.go
> 
> And on the server side the implementations of these interfaces
> 
>   https://gitlab.com/berrange/qapi-go-demo/-/blob/main/pkg/qapi/gen_server.go
>   https://gitlab.com/berrange/qapi-go-demo/-/blob/main/pkg/qapi/gen_server_events.go
> 
> 
> Finally we can consider what an implementation of a client looks like
> 
>   https://gitlab.com/berrange/qapi-go-demo/-/blob/main/cmd/sync-client/main.go
>   https://gitlab.com/berrange/qapi-go-demo/-/blob/main/cmd/async-client/main.go
> 
> And what an implementation of a server looks like
> 
>   https://gitlab.com/berrange/qapi-go-demo/-/blob/main/cmd/server/main.go
> 
> The key observation at the end is that the client/server impls are all
> using strongly typed APIs and don't have to concern themselves with JSON
> at all.
> 
> The two demo clients should work with an real QEMU or the demo server.
> The demo server won't work with qmp-shell, because I blindly assumed
> each JSON message had a terminating newlinue for ease of impl and this
> isn't compatible with qmp-shell. Solvable but I couldn't be bothered
> for this demoware.
> 
> 
> So what does this all mean wrt your series ? Not that much, probably
> just a little code deletion and some small tweaks.
> 
> 
> First, it illustrates that the approach taken for the Command / Event
> interfaces & the corresponding UnmarshallJSON/MarshallJSON methods is
> redundant code. Aside from that, I think everything else in your series
> around generating the basic types is essentially good.

Alright. The addition of Marshallers was done in the last
interaction. Before, I had the following for command and similar
to event:

    func MarshalCommand(c Command) ([]byte, error) 
    func UnmarshalCommand(data []byte) (Command, error)

Each of this series iterations had a more or less a different
idea of how the consumer of this generated code would use it, but
without a proper PoC on top, like you've done here and thanks for
that.

I'm happy with the suggestions and I'll incorporate them.
 
> Second, it shows a way to build on top of your series, to define a higher
> level API to make interacting with QMP much easier for app developers,
> eliminating all exposure of JSON & marshalling, in favour of formal APIs.
> This doesn't have to be done now, it can be a phase two, as long as we
> have an approximate idea of what the phase two API will look like, so
> we can validate the phase one design against these future plans.

Just want to put emphasis that the module that we will maintain
would still be considered alpha/unstable until we get the library
in a state we agree is good.

> NB What I've not especially considered in any of this is the impact of
> differing QEMU versions & their schema changes. The easy way out is to
> just re-run the generator for each version, putting them in a separate
> Go package. So you can do 'import' of "qapi/qemu/abi920" for 9.2.0
> schema APIs, or "qapi/qemu/910" for 9.1.0 schema APIs.

Yeah, plus what Markus said. Choosing qapi/qemu/900 should stay
working as long as QMP would guarantee (end of grace period of a
breaking change).

I'd consider sane Applications to bump the go module when they
bump the minimum required QEMU version for their supported use
case, to be able to handle new events and changes.

> The app dev would have to choose which version (or versions)
> they consume & implement against. Splitting versions across the
> whole package, avoids having to consider versioning within
> parameters of a single command/event.

Thanks again,
Victor
Re: [PATCH v3 0/8] qapi-go: add generator for Golang interfaces
Posted by Markus Armbruster 2 months, 2 weeks ago
Daniel P. Berrangé <berrange@redhat.com> writes:

> On Fri, Jan 10, 2025 at 11:49:38AM +0100, Victor Toso wrote:
>> I've pushed this series in my gitlab fork:
>> https://gitlab.com/victortoso/qapi-go/
>> 
>> The fork contains some tests, including tests that were generated from
>> QAPI's own examples from another generator created for testing, if you
>> are interested in it:
>> https://lists.gnu.org/archive/html/qemu-devel/2023-09/msg04946.html
>> 
>> I've generated the qapi-go module over each commit of this series, see:
>> https://gitlab.com/victortoso/qapi-go/-/commits/qapi-golang-v3-by-patch
>> 
>> I've also generated the qapi-go module over QEMU tags: v9.1.0, v9.2.0:
>> https://gitlab.com/victortoso/qapi-go/-/commits/qapi-golang-v3-by-tags
>
> Apologies in advance for what will be a long mail.....

I decline the opportunity for the pot to call the kettle black ;)

[...]

> NB What I've not especially considered in any of this is the impact of
> differing QEMU versions & their schema changes. The easy way out is to
> just re-run the generator for each version, putting them in a separate
> Go package. So you can do 'import' of "qapi/qemu/abi920" for 9.2.0
> schema APIs, or "qapi/qemu/910" for 9.1.0 schema APIs. The app dev would
> have to choose which version (or versions) they consume & implement
> against. Splitting versions across the whole package, avoids having to
> consider versioning within parameters of a single command/event.

I endorse this approach.

First, a practical point.  We want to generate bindings from the schema.
The generator we have works on the current version of the schema.  To
generate something spanning versions, we would have to make it work on
several versions, wouldn't we?  I don't think we can afford such
ambition.

Next, some supporting thoughts on interfaces and why bindings spanning
versions feels like way too much trouble to me.

QMP was designed for compatible schema evolution: we can evolve the
schema without breaking clients of the wire interface as long as they
stick to certain rules, such as "ignore unknown members in client
input".

Occasionally, the schema needs to change in ways that can affect such
clients.  We promise to give clients notice and time to adjust: we
formally deprecate the old interface, but keep it working for a grace
period, namely the release it was deprecated and one further release.

So, if your QMP client targets a specific version of the wire interface,
and avoids deprecated stuff, it'll be fine for at least two more
releases.  After that, you better upgrade the client.

This is a moderately flexible coupling between QMP client and server.

When you stretch it beyond the limit, things for the most part continue
to work.  Only the things we changed incompatibly can break.

If you need more flexible coupling, you can use QMP introspection to
target a *range* of wire interfaces.  The price is additional complexity
in the client.

Downstreams complicate this story a bit, but not in interesting ways, I
believe.

The existing C bindings are for the current schema version only.  That's
fine, because the only customer (QEMU) only cares for the current
version.

It would also be fine for external customers as long as they only need
the moderately flexible coupling described above.

Bindings spanning schema versions are significantly harder.

Do our rules for compatible schema evolution transfer from the
JSON-based wire interface to the bindings?  Not necessarily.  May
require acrobatics that make the bindings ugly.

Example: adding an argument to an event is compatible evolution.

Adding it to the C function to send the event is not compatible: it
breaks the build, and all callers need to be adjusted to fix it.  Yes,
this may not be an issue with certain other languages, but my point
stands: not necessarily.

Example: adding a member to an object type is compatible evolution.

Adding it to the type in the ABI is not compatible: mixing ABI before
and after won't link if you're lucky, and run amok at run time if you're
not.