-
Notifications
You must be signed in to change notification settings - Fork 55
[Versioning]How to modify workflow code without breaking changes
So there are three types of “breaking changes” to discuss here:
- Non-deterministic errors – this won’t happen in iWF user application workflow
- Business in-compatible changes
- Technical in-compatible changes (sorry, naming is very hard here)
First of all, there are no “non-deterministic-errors” in iWF user workflow code because iWF workflow is not running in a “replay model” like Temporal does. The iWF workflow code is serving as RESTful APIs, meaning that the modification can take effect immediately once deployed, for current and future states.
For example, if we have a workflow with these 4 steps/states in sequence
And we want to add a new step/state before S3, then the code will then become
This is not safe to do in native Temporal workflow applications, because it breaks the replay determinism. But this is totally safe in iWF:
- For any workflow execution that is already in S3 or after, it will continue without problems
- For any workflow execution that is in S2 or before (both existing and new), it will use the new flow – execute S31 before S3.
Then we move to the second type: business in-compatible changes. For the above example, what if we want the existing workflow executions to keep using the old flow, if it started with the old one?
Essentially, the workflow logic should become:
To implement such a behavior, the user workflow can just set a custom flag/version in the startingState of the workflow. Update the flag/version value when updating the workflow code. Then based on the flag/version to make the decision on going to S31 or S3 like in the above diagram.
Finally, maybe the most common breaking changes are technical in-compabible changes. These are exactly the same as all the API breaking changes in microservices, because the iWF workflow code is serving as RESTful APIs for iWF server callback. The callback will fail if the API is “in-compatible”:
- Rename or remove a Workflow/WorkflowState, but the Workflow/WorkflowState is still running
- Making breaking changes to any data model in the workflow, while the data model is still being used with old data values. E.g. change a field from optional to required, rename a field, add a new required field, etc.
Essentially, users should keep in mind that all the workflow code is running as a RESTful service, and use the standard approaches to avoid those breaking changes.
Some tips/best practices, for examples:
- In Java, using Optional to add a new field is always safe when changing data models
- In Java, Make sure to set “ignore unknown fields” in Jackson (default in iWF already) for removing old fields
- Use duplicated Workflow/WorkflowState when renaming, and remove the duplicated after all old workflows are completed.
- By default, the WorkflowType and StateId are the class simple names. But you can override the method to use a different name to make it different from the class name.
- Use system search attributes “IwfExecutingStateIds” and “IwfWorkflowType” to check if there are any old workflows running, when rename/removing the old StateId or WorkflowType. NOTE: IwfExecutingStateIds is disabled by default because of cost. To enable it by setting "disableSystemSearchAttributes" to false for WorkflowConfig when starting a workflow. We are working on this as a more cost effective way – it only records the SearchAttribute for states that have waitUntil. And also comes with several optimizations.