-
Notifications
You must be signed in to change notification settings - Fork 33
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
Warn on unfinished workflow handlers #294
Conversation
"handler attribute: " + | ||
"`[WorkflowSignal(UnfinishedPolicy=HandlerUnfinishedPolicy.Abandon)]`. The " + | ||
"following signals were unfinished (and warnings were not disabled for their " + | ||
"handler): {Handlers}"); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Shouldn't the handlers be returned as the own key and the message be constant? That way the message can be easily filtered for and any unfinished updates could be easily extracted? Basically I am asking why we can't treat this like other SDKs logs that use structured logging
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Can you clarify what you mean by constant? This doesn't change if that's what you mean.
We did want to put the literal linked list of this information in the structured logger, but .NET forces linked list formatting and I am matching the Python PR by using JSON. I considered making a whole new data type, but then I would need to make the data type available as part of our public API and guarantee compatibility on it. We are not doing that in the Python PR (there is no extraction of handlers from the warning, though it could be easily done). .NET wasn't meant to add any features over the Python one, which is English message only w/ JSON appended. If we do want to expose data types for handlers we can, but I think that requires more thought and more API surface and more guarantees.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The message here has workflow ID substituted in no Workflow {Id} finished
? That makes it harder to filter for occurrences of this log in a logging system since users can't just filter on a direct match.
Not sure you need to make the data type public, as long as the json format is consistent it could be parsed. You do have an UpdateInfo
type maybe you could reuse that? I don't feel strongly about requiring the list to always be formatted in json, I would expect the format to be left to the logger, but if others do that's fine.
For example when we warn on unhandled signals in Go we return the list as part of a key-value
. For updates I would do something similar (although we would also need to add UpdateID as well)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The message here has workflow ID substituted in no Workflow {Id} finished? That makes it harder to filter for occurrences of this log in a logging system since users can't just filter on a direct match.
This is in addition to scoped values. The thing with .NET unlike other SDKs is that their scopes are often not logged. So we also added workflow ID to the message here. Arguably I should add workflow ID to all English messages from a workflow, but regardless, all workflows have a "scope" with a dictionary with this information and the logger can be configured with a format to expose this scoped information (but many don't, and it's not in there by default like it is in Go and Python).
would expect the format to be left to the logger, but if others do that's fine.
In .NET, there is not a good stable/format for list/array items. It is more important to make the English read well than to reuse .NET collection types in the structured log. So we can either do string or new data types w/ custom formatter. I think if users get to the point where they want to extract the structure out, we can expose entirely new, stable data types that reflect this structure (a list of two-field objects), but until we want to expose those data types, we can/should stick with string. And if/when we do want structured objects, we should also add them to the Python warning.
For example when we warn on unhandled signals in Go we return the list as part of a key-value. For updates I would do something similar (although we would also need to add UpdateID as well)
This is more than a list, this is a list of objects with two fields each. The question becomes whether we need a stable data structure for that. I don't think we do. For instance, when Go SDK has the code for this, you will likely need to choose between a new object to get formatting right (that users probably can't access unless we're talking exposure), or JSON formatting. Though Go may be lucky enough to have a logger that defaults to a reasonable format for lists, but .NET and others don't necessarily.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I am going to refactor to a new data type that I just won't expose. Users won't really be able to do anything with it except via reflection (but if we decide to expose this data type later we can).
EDIT: There is now a new data type but users can't do anything with really besides reflection. We can't make it extend/implement anything useful like collection, because .NET formatting is short-circuiting that in its formatting logic.
{ | ||
if (warnableExecution is { } toRemove) | ||
{ | ||
warnableInProgressSignals.Remove(toRemove); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Note that this is just going to remove the first entry that has the same name (since signal name is not unique among handler executions). So using AddLast
above is arguably misleading, since the order loses meaning via this call. A MultiSet
(i.e. collections.Counter
in Python might be a good choice for in-progress signals here. In my PR I gave them a unique ID but perhaps that was unnecessary.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Note that this is just going to remove the first entry that has the same name (since signal name is not unique among handler executions)
Not true, note that toRemove
is a LinkedListNode
. It has nothing to do with the contents of the node. This collection was specifically chosen for the cheap node removal by node that .NET offers.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ohhh, right. Yep I was thinking warnableExecution
was the string signal name.
private static readonly Action<ILogger, string, WarnableUpdates, Exception?> UpdateWarning = | ||
LoggerMessage.Define<string, WarnableUpdates>( |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Could maybe combine these two warning functions generically to save on duping the boilerplate part of the message. Not a big deal.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
👍 In this case I am matching the contents of temporalio/sdk-python#556, but can make this change generally if needed.
What was changed
Workflow.AllHandlersFinished
bool propertyHandlerUnfinishedPolicy
enumUnfinishedPolicy
option to[WorkflowSignal]
and[WorkflowUpdate]
attributes and defaulted toWarnAndAbandon
WorkflowSignalDefinition
andWorkflowUpdateDefinition
classes (used by manual handler registration)Checklist