Get the source code for this video here → the-dotnet-weekly.ck.page/masstransit-saga Want to master Clean Architecture? Go here: bit.ly/3PupkOJ Want to unlock Modular Monoliths? Go here: bit.ly/3SXlzSt
The most clear crystal video I've seem about MassTransit Sagas. This video must referenced on masstransit documentation. Great video as well. Please, bring us a video about the MassTransit Courier Service, it is great for distributed transactions with compensation routines!
I think it would've been even better if I had used separate services for each handler. But I felt that would overcomplicate the matter and distract the user from the actual Saga pattern. I'm glad that you enjoyed this one :) P.S. Thanks for the video suggestion!
@@MilanJovanovicTech It would had been far better, if handlers would had been split in to 3 different services. It somewhat simulates the real world applicatoin, just my thoughts.
Great video. This concept feels like "do I really need this? Can't I create an entity and update it with its methods instead?" But I think if you make another video about implementing a complex flow with traditional ways and saga pattern in the same video and show differences, people will have an "ahaa" moment
@@MilanJovanovicTech There are two big weaknesses with sagas. One is the expertise required to write and maintain them well. The other is that they can never plug all the data leaks: There are always edge cases where data gets lost. The workflows are partly non-deterministic because they are weakened by issues such as latency, plus the workflows tend in some cases to never complete such as with timeouts.
@@stephendgreen1502 if you end up with "data leaks" in your system, it's a result of poor programming, not message loss. There are many companies using extensive sagas with MassTransit that have no lost transactions, and they're doing 400,000 an hour in real-time. So, your experience may vary, but blaming the technology/pattern is misguided.
I would definitely pay for a MassTransit course. Chris's videos are great, but would love some more real examples of Routing Slips (Choreography) and State Machines (Orchestrators). State/Status/Processing steps are one of my biggest struggles. Would love to know more using examples like Invoices/Reserving Inventory/Billing Customer/Shipping Items/etc. Also would be helpful if you explained how to correlate using an integer as an Id.
@MilanJovanovicTech What would be the process if you wanted to add an additional step in your saga after it is already in use? when the new code is published will messages automatically move to the new state added? or should that be avoided and once a saga is deployed it shouldnt be modified?
Any Saga that goes past the old state probably won't be updated. But Sagas in the middle, that can pick up the new messages (and new Sagas of course) should handle that case. This is actually a brilliant idea for a new video.
Awesome tutorial. I have one question. At 14:34, we see that you landed in the last step, which is OnboardingCompletedHandler. In this step, the Consume method hasn't finished yet. When we take a look at the saga state in DB at 14:41, it is at the Final state, which I assume that the saga has finished. But we haven't still complete the Consume method in the last step. What would happen if the last OnboardingCompletedHandler throws an error? Will the saga state be still Final, how can we detect such issues where the saga seems complete but actually the last step erred. Thank you.
"What would happen if the last OnboardingCompletedHandler throws an error? Will the saga state be still Final, how can we detect such issues where the saga seems complete but actually the last step erred" - Yes in this case the Saga would transition to the final state. We can add another step, where we wait for a message while in the Onboarding state. - During(Onboarding, ...) I'll have to do a separate video on error handling, but essentially MassTransit allows you to handle faults whenever you publish a message.
This is what I'm looking for with a POC I want to develop that allows users to define jobs in which one "consumer" is ran after the previous consumer's condition has been met. The only problem is that I need something that is dynamic since the user would be defining these states via UI, is that possible?
@@MilanJovanovicTech Basically want to create an application where users can define data flows via UI that will allow them to specify logic gates on incoming stream data. I did more digging into it and it could probably be defined with sagas but I think Apache Flink is probably a better choice
Apologies if I missed it but is there a reason why in your example you're "publishing" the command messages rather than "sending" them to a specific consumer? Publishing certainly works and involves less setup but I'm wondering if ultimately it's better to use send rather than publish for clarity of intent.
Hi Milan, i am trying to use sql instead of postgres to persist the state entity. If i run the saga using the in memory repository all works but if i use entity framework with sql it does not run. I guess some issue in the entity framework settings. Any suggestion how to properly configure entity framework using sql database? Thanks and keep up with the great work!
Hello Milan, I have a question if I need to build a saga between two microservices from A (with its database) to B (with its database) the saga in a clean architecture where should I write it? In microservice A or B? Thanks
Hi Milan, first of all - I like your tutorials very much. There are many of them which helps me out. I have a question: does the saga continues if let's say the process getting stopped between the Welcome and the FollowUp state? If so, does this work automatically or have one to send a message to trigger continuation?
EDIT: Nvm, forgot the AddConsumers call in the setup. ---- Do you need to start the bus or have a hosted service running or something? If I try this, I don't hit the breakpoint after using bus.Publish, however there are no errors or anything in my logging. Just nothingness.
I don’t know what it is, but for some reason this feels “off”. Just can’t put my finger on it. You have the regular publish/subscribe handlers which all just accept a command and send out a new one, but then you have one central state machine which accepts those and then sends the new publish command. I guess it feels off because it is all an 1-on-1 for the state and commands. You’re pretty much just resending the commands because of that. I guess this becomes a lot more useful with more complex/involved flows and it’s easier to track and continue flows in this way. I guess you made it too simple to follow along 😊
I have to make these example simple so that they're easy to demonstrate. Think of an order processing flow online. You have a few steps: - Placing an order - Reserver items on stock - Making the payment for the oder - Shipping the order to the customer This is difficult to implement as one long-lived transaction. Which is why we can introduce a saga (state machine) to manage the individual steps. Moreover, these steps can be handled by different physical services which makes it even harder to track. So a central orchestrator that can tell you the current state of the order processing flow is a unique advantage.
We have a few options, mainly we want to retry processing the step so we can complete the saga. However, if we can't continue, we can publish a "failure" message and consider how we can reverse the effects of previous steps. Like implementing an "undo" operation.
Could you please add flow diagram when you explain the new concept it will better for us to understand more. No one follows the entire code unless they work. Please explain flow diagram and highlight any extra things need to be noted.
Get the source code for this video here → the-dotnet-weekly.ck.page/masstransit-saga
Want to master Clean Architecture? Go here: bit.ly/3PupkOJ
Want to unlock Modular Monoliths? Go here: bit.ly/3SXlzSt
The most clear crystal video I've seem about MassTransit Sagas. This video must referenced on masstransit documentation. Great video as well. Please, bring us a video about the MassTransit Courier Service, it is great for distributed transactions with compensation routines!
I think it would've been even better if I had used separate services for each handler. But I felt that would overcomplicate the matter and distract the user from the actual Saga pattern. I'm glad that you enjoyed this one :)
P.S. Thanks for the video suggestion!
@@MilanJovanovicTech It would had been far better, if handlers would had been split in to 3 different services. It somewhat simulates the real world applicatoin, just my thoughts.
Great video. This concept feels like "do I really need this? Can't I create an entity and update it with its methods instead?" But I think if you make another video about implementing a complex flow with traditional ways and saga pattern in the same video and show differences, people will have an "ahaa" moment
That's a great suggestion for a future video!
The best topic you have covered, in my opinion. Very much appreciated.
Now I need to move these handlers into separate services, and add distributed tracing on top.
@@MilanJovanovicTech And hire some code-skilled support engineers to deal with data lossiness.
@@MilanJovanovicTech There are two big weaknesses with sagas. One is the expertise required to write and maintain them well. The other is that they can never plug all the data leaks: There are always edge cases where data gets lost. The workflows are partly non-deterministic because they are weakened by issues such as latency, plus the workflows tend in some cases to never complete such as with timeouts.
@@stephendgreen1502 if you end up with "data leaks" in your system, it's a result of poor programming, not message loss. There are many companies using extensive sagas with MassTransit that have no lost transactions, and they're doing 400,000 an hour in real-time. So, your experience may vary, but blaming the technology/pattern is misguided.
I would definitely pay for a MassTransit course. Chris's videos are great, but would love some more real examples of Routing Slips (Choreography) and State Machines (Orchestrators). State/Status/Processing steps are one of my biggest struggles. Would love to know more using examples like Invoices/Reserving Inventory/Billing Customer/Shipping Items/etc. Also would be helpful if you explained how to correlate using an integer as an Id.
Noted! Maybe I can do a longer YT video first, and see how that goes.
I find Chris' videos are hard to follow for beginner
Great! Milan Now I am waiting for a video where you explain a video on how to handle an undo/rollback operation
Here's the conceptual approach: ua-cam.com/video/FPVzevl6Ri8/v-deo.html
Awesome content. I appreciate the effort you put into your videos.
Glad you enjoy it! Thanks :)
Thanks for valuable content, Milan.
Most welcome!
@MilanJovanovicTech What would be the process if you wanted to add an additional step in your saga after it is already in use? when the new code is published will messages automatically move to the new state added? or should that be avoided and once a saga is deployed it shouldnt be modified?
Any Saga that goes past the old state probably won't be updated. But Sagas in the middle, that can pick up the new messages (and new Sagas of course) should handle that case. This is actually a brilliant idea for a new video.
Awesome tutorial. I have one question. At 14:34, we see that you landed in the last step, which is OnboardingCompletedHandler. In this step, the Consume method hasn't finished yet. When we take a look at the saga state in DB at 14:41, it is at the Final state, which I assume that the saga has finished. But we haven't still complete the Consume method in the last step. What would happen if the last OnboardingCompletedHandler throws an error? Will the saga state be still Final, how can we detect such issues where the saga seems complete but actually the last step erred. Thank you.
"What would happen if the last OnboardingCompletedHandler throws an error? Will the saga state be still Final, how can we detect such issues where the saga seems complete but actually the last step erred"
- Yes in this case the Saga would transition to the final state.
We can add another step, where we wait for a message while in the Onboarding state.
- During(Onboarding, ...)
I'll have to do a separate video on error handling, but essentially MassTransit allows you to handle faults whenever you publish a message.
🫡🫡🫡Hatsoff to you and efforts with quality content that you create.
Most welcome! :)
This is what I'm looking for with a POC I want to develop that allows users to define jobs in which one "consumer" is ran after the previous consumer's condition has been met.
The only problem is that I need something that is dynamic since the user would be defining these states via UI, is that possible?
Looks like some sort of dynamic saga generator?
@@MilanJovanovicTech Basically want to create an application where users can define data flows via UI that will allow them to specify logic gates on incoming stream data. I did more digging into it and it could probably be defined with sagas but I think Apache Flink is probably a better choice
It would be great if you cover also how to rollback if some state we cannot move forward? Is there some builtin functionality also in MassTransit?
I did, albeit not with MassTransit: ua-cam.com/video/FPVzevl6Ri8/v-deo.html
super good explanation and nice video edit. perfect
Thank a lot!
Great video, as always :)
Thank you! Cheers!
Thanks Milan. It's very useful video.
Glad it was helpful!
Apologies if I missed it but is there a reason why in your example you're "publishing" the command messages rather than "sending" them to a specific consumer? Publishing certainly works and involves less setup but I'm wondering if ultimately it's better to use send rather than publish for clarity of intent.
Technically speaking, you're right on this. Send is the appropriate one to use for commands. I just find publish more convenient.
Hi Milan, i am trying to use sql instead of postgres to persist the state entity.
If i run the saga using the in memory repository all works but if i use entity framework with sql it does not run.
I guess some issue in the entity framework settings.
Any suggestion how to properly configure entity framework using sql database?
Thanks and keep up with the great work!
I'd check the MassTransit docs, and after that just start debugging.
Do you see the tables in the DB? Maybe you need to create another EF migration?
Hi Milan, creating a migration fixed the problem. Thanks a lot!
I would love if you would go in depth on this topic more. Not very many well made MT vids online.
Can do!
What a legend!
Much appreciated!
Hello Milan,
I have a question if I need to build a saga between two microservices from A (with its database) to B (with its database) the saga in a clean architecture where should I write it? In microservice A or B?
Thanks
Which one should control the flow?
@@MilanJovanovicTech Microservice A
Hi Milan, first of all - I like your tutorials very much. There are many of them which helps me out. I have a question: does the saga continues if let's say the process getting stopped between the Welcome and the FollowUp state? If so, does this work automatically or have one to send a message to trigger continuation?
Yes, if you use a saga persistence mechanism that isn't in-memory. The saga stores the current state, so it can pick up where you left of.
EDIT: Nvm, forgot the AddConsumers call in the setup.
----
Do you need to start the bus or have a hosted service running or something?
If I try this, I don't hit the breakpoint after using bus.Publish, however there are no errors or anything in my logging. Just nothingness.
Glad you have it sorted 😁
Great video, thanks for sharing
You bet!
Ultimate vedio about saga
💪
sos un crack chamo
Why did you declare your commands as records but not your events?
Easier serialization
I don’t know what it is, but for some reason this feels “off”. Just can’t put my finger on it. You have the regular publish/subscribe handlers which all just accept a command and send out a new one, but then you have one central state machine which accepts those and then sends the new publish command. I guess it feels off because it is all an 1-on-1 for the state and commands. You’re pretty much just resending the commands because of that. I guess this becomes a lot more useful with more complex/involved flows and it’s easier to track and continue flows in this way. I guess you made it too simple to follow along 😊
I have to make these example simple so that they're easy to demonstrate. Think of an order processing flow online. You have a few steps:
- Placing an order
- Reserver items on stock
- Making the payment for the oder
- Shipping the order to the customer
This is difficult to implement as one long-lived transaction. Which is why we can introduce a saga (state machine) to manage the individual steps.
Moreover, these steps can be handled by different physical services which makes it even harder to track. So a central orchestrator that can tell you the current state of the order processing flow is a unique advantage.
Isn't saga about compensating?
Why is a step fails and you need to roll back?
what happens when we face an error during the steps?
We have a few options, mainly we want to retry processing the step so we can complete the saga. However, if we can't continue, we can publish a "failure" message and consider how we can reverse the effects of previous steps. Like implementing an "undo" operation.
To a compensation
@@MilanJovanovicTech would be really interesting a video on how to handle an undo/rollback operation
@@emanuelrodriguez3155 Tottaly agree, in masstransit it is called courier service to handle these cases...
Could you please add flow diagram when you explain the new concept it will better for us to understand more. No one follows the entire code unless they work. Please explain flow diagram and highlight any extra things need to be noted.
Yes
I don't know much about the roles of RabbitMq and Postegresql
Research
I literally been watching tutorials about this last week
Cool. Mind sharing a few that you watched? I'd love to see it
Thank you
You're welcome!
I applaud for the effort made for the vide but there are too many topics to explain in a short vide. it is not detailed enough for begineers.
This isn't really a beginner-friendly topic
Your videos are getting more complicated 😢
The topic is complicated...
This is a very complicated stuff. Do you think that it is a good idea to use Redis to store SAGAs instead of Ef database Milan?
Maybe, if Sagas are short-lived. You can always switch it with a SQL DB