Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

add S2 Message Union type #37

Open
wants to merge 4 commits into
base: main
Choose a base branch
from
Open

add S2 Message Union type #37

wants to merge 4 commits into from

Conversation

victorgarcia98
Copy link
Collaborator

Recently, I had the need to wrap the S2 messages in an envelope to simulate accelerated interactions between a CEM and a RM.

Nonetheless, for this to work, the message_type had to be a Literal. This PR refactors the code to use Literals instead of regular fields.

Example usage:

from pydantic import BaseModel, Field
from typing import Union, Annotated,Type, Literal
from datetime import datetime

from s2python.message import S2Message

class MetaData(BaseModel):
    dt : datetime

class S2Wrapper(BaseModel):
    message : S2Message = Field(discriminator='message_type')
    metadata : MetaData

Signed-off-by: Victor Garcia Reolid <[email protected]>
@victorgarcia98 victorgarcia98 added the enhancement New feature or request label Mar 21, 2024
@victorgarcia98 victorgarcia98 self-assigned this Mar 21, 2024
@lfse-slafleur
Copy link
Member

@victorgarcia98 Apologies for a late response. I do not work at TNO anymore and it took some time to get settled in my new situation again. Reviewing now!

@@ -0,0 +1,12 @@
import inspect
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We already have a version of this file in generate_s2_message_type_to_class.py to generate a mapping from message_type literals into the appropriate class. You are using it to generate a list of all message types into an alternative S2Message type which only contains messages.

I think we can combine the 2. My proposal would be to move TYPE_TO_MESSAGE_CLASS to the new src/s2python/message.py file and generate this entire file with a single script. The result would be that we generate both the TYPE_TO_MESSAGE_CLASS constant as well as the S2Message type.

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Is it correct that this suggestion is purely a refactoring request?

Since I'll be taking over this PR on behalf of @victorgarcia98, and I'm still a little uncertain as to increase the scope of this PR's code changes, would it be okay to tackle this refactoring in a separate PR?

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

That is not an issue at all, lets do it in a seperate PR. It is an idea purely for refactoring/code deduplication. If you are busy, I can submit the PR for review to you.

SessionRequest
)

S2Message = Union[
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This name S2Message is already used here:

class S2Message(Generic[C], ValidateValuesMixin[C], BaseModel):

My proposal is to change the original usage from S2Message to something like S2MessageComponent as it is more descriptive and keep S2Message as you introduce it here. Especially since the members of this Union type are all the messages and not parts of a message.

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Good idea, will rename.

Copy link
Member

@lfse-slafleur lfse-slafleur Aug 29, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks! 😄 Will keep this open so you can optionally resolve it. (I usually keep the conversations open until the commit is made to keep an overview, but feel free to use it however you want of course)

@@ -214,7 +214,7 @@ class Config:
..., description="ID of this instruction (as provided by the CEM) "
)
message_id: ID = Field(..., description="ID of this message")
message_type: str = Field("InstructionStatusUpdate", const=True)
message_type: Literal["InstructionStatusUpdate"] = Field(default="InstructionStatusUpdate")
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

No issues with moving from a pydantic const to the Literal type. However, datamodel-codegen does not support to use it for pydantic v1 output. Prioritising updating to pydantic v2 so it is automatically generated. #10

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

PR is ready that has taken care of generating Literals instead of constants by moving to pydantic v2: #40

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

HomeAssistant is not yet compatible with Pydantic 2. We are tracking home-assistant/core#99218 to see when it is, but until then I'm not sure of the implications with regards to compatibility if the s2-python package would already migrate to Pydantic 2. Is the use of a Literal problematic at this point, or do you only regard it as a less preferred option over moving to Pydantic first.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

HI @Flix6x ! gen_s2.py is the output of the datamodel-codegen code generator. As such, we can move to literals but in its current shape it would be modifying a generated code file manually or as a post-processing step.

Perhaps we should split the discussion. We could discuss 'Why and when to move to pydantic v2' here: #10

Then we can keep the discussion in this comment thread to how to close this PR. Moving to pydantic v2 solves this automatically as datamodel-codegen uses the Literal type in its output for pydantic-v2 output. So I was expecting that moving to pydantic v2 would solve this issue without any extra work. However, if we want to move to Literal without moving to pydantic v2, I would like to propose an alternative. Manually altering generated files is error prone. We could move the Literal lines from the generated code that are introduced by this PR to the non-generated classes, essentially overwriting the fields as we have done with ID's. The referenced line of this comment refers to InstructionStatusUpdate. An example on where to move it (see the last line):

File: https://github.com/flexiblepower/s2-python/blob/main/src/s2python/common/instruction_status_update.py

@catch_and_convert_exceptions
class InstructionStatusUpdate(
    GenInstructionStatusUpdate, S2Message["InstructionStatusUpdate"]
):
    class Config(GenInstructionStatusUpdate.Config):
        validate_assignment = True

    message_id: uuid.UUID = GenInstructionStatusUpdate.__fields__["message_id"].field_info  # type: ignore[assignment]
    instruction_id: uuid.UUID = GenInstructionStatusUpdate.__fields__[
        "instruction_id"
    ].field_info  # type: ignore[assignment]
    message_type: Literal["InstructionStatusUpdate"] = Field(default="InstructionStatusUpdate")  # <-- Line which is added

What do you think of this idea? In case you like it, perhaps its a good idea to first test with a small case to see if it has the intended effect.

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I like your idea. Not meddling with the generated code seems like a sensible policy.
I tried out your suggestion and then ran datamodel-codegen --input specification/openapi.yml --input-file-type openapi --output src/s2python/generated/gen_s2.py to see whether gen_s2.py would contain Literal message types, but they are still str.
Was this what you had in mind, or am I missing a step?

(As of today, Home Assistant is still not compatible with Pydantic 2.)

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Originally what I had in mind was to manually overwrite the types that datamodel-codegen generate with manually added lines such as message_type: Literal["InstructionStatusUpdate"] = Field(default="InstructionStatusUpdate") # <-- Line which is added. However, as you are taking over pydantic v1 branch, feel free to use another method which fits for your efforts.

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

After consultation with @victorgarcia98:

  • I understand better now that no changes on the generated classes were actually expected, because we are now overwriting the message type in their sub-classes.
  • I realized I had been too enthousiastic with overwriting message types in this PR, because I had unknowingly also added a message type to classes that were not themselves S2 messages, but rather described elements contained in such messages.

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

To move this PR forward, I propose I remove all changes relating to the Literal issue and let it only contain the addition of the S2Message class and the renaming of the old S2Message class into S2MessageComponent.

In #56, I propose to take ownership of releases from the pydantic-v1 branch, using dev versioning.

@Flix6x Flix6x self-assigned this Aug 29, 2024
# Conflicts:
#	src/s2python/generated/gen_s2.py
…ut-file-type openapi --output src/s2python/generated/gen_s2.py`

Signed-off-by: F.N. Claessen <[email protected]>
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
enhancement New feature or request
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants