Couldn't agree more. I'd be annoying to keep explaining to new people in the project the meaning of this obfuscated code. I would pick a bit more lines of clear code over shorter, but much more complex version every time.
The point is to understand the code without explanation. This is hard to read especially for begginers and interns even for mid developers. I still prefer the first version of the code.
That's just because Monads and FP in general are almost always ignored when learning programming. There's nothing intrinsically hard to understand about this approach (albeit monads can be tricky to grasp), you just need to get familiar with it like with everything else
We have been discussing this very thing at work and I really like your solution here. I’ve a feeling we will be using exactly this in our controllers from now on. Thanks for the video, it’s excellent!
While this is nice looking and certainly functional, i think it's not a major improvement over the controller code you showed at the start of this video. Imo, the original version does comunicate intent way better than the new version does (specifically the bind portion is something that someone looking at this code for the first time will have to investigate before they are able to understand what is actually happening). Having the match function to decide on a handler to call given a certain result type ( success vs. failure) is nice, but i would personally leave it at that and keep the rest of the code as it was. Anyways: Thanks for showing of the more functional approch, definitely an intersting way to do it !
Looks nice and fancy afterwards, but by wrapping everything in static extension methods I would argue it is harder to read, debug and unit test. The original way was clean, simple, effective and easily tested.
@@Cruize91 Why is it rude? The main critics of this video (and any video I create about functional programming) are people who don't do any FP. They see everything through an OOP lens, and can't admit that pure functional code cam indeed be clean, simple, effective, and easily tested. Oh, and don't judge a person based on one comment... 😅
@@MilanJovanovicTech I didn't judge your person at all. I criticized the response. Secondly, you're putting words in my mouth. I didn't say 'rude'. You have two commenters here who are genuinely trying to interact with you and your content. The first comment was simply a voice of not understanding. You could've handled that so much better by replying with a friendly stock answer, if this is something that occurs so often. Secondly, I'm simply stating that I disagree with this kind of response, because aren't we here to learn? A guy comments with some genuine concerns, so it's a chance to educate nicely. I'm not saying this to scold you, I'm just saying that I like your content, and would like to keep following it, but this kind of behavior is gonna turn me away.
Only because most people are not used to FP style code. Otherwise, this is also easy to read, and if anything it's even easier to read, since it follows a very natural flow from step to the other.
Milan Nice Explanation. One Suggestion , like you're using already created project for Video. I think it's hard for viewers to understand when you start coding in already created classes etc. I think it will be good if you demo it in sample small project. it will be easy to understand. :)
You've created a Result monad. This comes out of the box in most functional languages. It's such a round-about way of doing it in C# that I usually just don't bother writing all of the "extra code to make it more F#'y". The Result class makes it possible to make a distinction between an expected (validation / domain) error and an actual unexpected exception (e.g. the database is down or we accidently introduced a bug).
How do you not have to write authentication scheme on the controller function? Which one is picked as default? If you only have one active, will it be picked as default?
On a side note, what is the benefit of using that pattern, is it simply to adhere to the "Clean architecture setup"? Performance wise, how is this better/worse than simply having some interface IService with method DoSomething(), and then injecting that service into the application? Of course I think the way you do it in this video is nice, but it also requires a little more boilerplate to get running. And lastly, thanks for the videos, I love them :)
Performance wise it's slightly worse, because of more allocations and method calls. But FP is very nice to work with, although it takes some time getting used to. The end result is the same. I'm just trying to raise awareness about FP.
We receive REST payload object to create another request inside controller and process it somewhere else? Well... It's far better to encapsulate logic in minimal API or FastEndpoints than using another layer.
@MilanJovanovicTech, that is a super clean and easy to read design, thanks for your great ideas and inspirations. Don't pay to much attention to ones who say it's complex and unreadable. Even for average mind as mine it is quite straightforward (even given I have started to learn c# dotnet a few month ago). I could only think of better name for Bind, like Execute, but probably you have some convention as I see you always do in dotnet)
I have a few more videos on ROP, so if you're up for it check them out: - th-cam.com/video/dDasAmowFts/w-d-xo.html - th-cam.com/video/zuy2j8vxgYc/w-d-xo.html - th-cam.com/video/vGkgsduwnc4/w-d-xo.html
Hi Milan. As much as I love the extension methods and the flexibility and clean code practices they bring to the codebase I'm skeptical of their impact on performance/memory. Static classes gets instantiated as soon as the application starts and they live in heap memory throughout application's lifetime. I'm not sure if using too many static classes/methods is going to have a significant impact on application's performance as it will cause frequent GC triggers? Correct me if I'm wrong please. Thankyou.
@@MilanJovanovicTech That's where Nick and you might differ I guess. But I agree most clients are either unaware of or simply don't care about memory efficiency and performance. They just want working solution with minimal overhead.
Love this with the ErrorOr library, have something very similar in place currently. In my opinion throwing Exceptions for errors which are expected (we know about) is logically incoherent since they are not exceptional in nature. What's your take on Errors vs Exceptions?
I think that instead of creating additional abstraction with those Bind and Match methods it could be simplified just by not going back to the controller with IsFailure, Value, Errors properties, instead returning the actual result (value) from the query and just a Task from command, and if there is a validation/business logic error - just thrown an exception with proper context and handle it to returned desired code & message in this way in the controller it would be just a 2 liner:
@@MilanJovanovicTech let's assume that you send a command that violates some business rules that are checked in the domain layer, in this approach instead of throwing an exception, you have to pass information up from the domain layer about whether it failed or not, thus coupling domain, application and presentation layers by branching the 'IsFailure' property
Hi Milan, nice and efficient work, thanks for sharing it! I see you are using CancellationToken parameter in your Post Methods. I heard that we should not be using CancellationToken in our Post Methods because of in case of cancel there may be an inconsistency in the data as a result of the process being interrupted. What do you think about this?
Can you please make some videos on aws cloud formation + azure pipeline using .Net core , Wanted detail video how API triggers lambda through cloud formation.
Hello Millan! Thanks for the great new video! I always share your videos with my friends in Brazil :) So I have a question, what do you think about adding a mapper from your DTO to Commands/Queries? like AutoMapper. I think your controller will have less lines of code, is that a good idea?
Prefer/try explicit (or implicit) operators to your DTOs to do the mapping. Yes, you have to write the mapping code yourself, but that's a good thing, especially when it comes to static analysis and debugging.
ปีที่แล้ว +1
You can try mapperly, it uses source generators to create static mapping. So at any point you can remove it, and keep the code. It also does not hide references behind reflection. Drawback is that it does not support everything that Mapster does, ex. projections.
Good work, every video we enjoy and learn something new & interesting, but what if think about time complexity, we usually try to make our codes more simple making reops or like this stuff, but is it suitable way keeping DATA-STRUCTURE/ALGORITHMS/TIME-COMPLEXITY/ in mind.... please guide. Thanks.
Interesting approach, but I wonder why go to all this trouble when you'd be able to achieve arguably a similar result using Minimal APIs. Also you really should have validation in there somewhere...
I don’t see how any of the code in a minimal api would be different/simpler at all. You would just be using Results.* instead of ActionResult. The body of a minimal API and a controller action are identical, they are functions
Want to master Clean Architecture? Go here: bit.ly/3PupkOJ
Want to unlock Modular Monoliths? Go here: bit.ly/3SXlzSt
Why do you not create any Course about c#?
@@naldiojoaquim8920 Still too early to do that, in my opinion :)
Definitely clean, but harder to understand for average developers. I would prefer somewhat balanced simplicity.
I'd say a quick explanation will get anyone up to speed
I agree. Also about the same amount of lines in the controller but more complex and rigid.
Couldn't agree more. I'd be annoying to keep explaining to new people in the project the meaning of this obfuscated code. I would pick a bit more lines of clear code over shorter, but much more complex version every time.
The point is to understand the code without explanation. This is hard to read especially for begginers and interns even for mid developers. I still prefer the first version of the code.
That's just because Monads and FP in general are almost always ignored when learning programming. There's nothing intrinsically hard to understand about this approach (albeit monads can be tricky to grasp), you just need to get familiar with it like with everything else
We have been discussing this very thing at work and I really like your solution here. I’ve a feeling we will be using exactly this in our controllers from now on. Thanks for the video, it’s excellent!
Check out the OneOf library, it has this built in
While this is nice looking and certainly functional, i think it's not a major improvement over the controller code you showed at the start of this video. Imo, the original version does comunicate intent way better than the new version does (specifically the bind portion is something that someone looking at this code for the first time will have to investigate before they are able to understand what is actually happening). Having the match function to decide on a handler to call given a certain result type ( success vs. failure) is nice, but i would personally leave it at that and keep the rest of the code as it was.
Anyways: Thanks for showing of the more functional approch, definitely an intersting way to do it !
I think if more people talked about FP it wouldn't be so confusing to people. But Bind is a straightforward concept, as is anything in FP honestly
How do you know that linq Where returns a filtered list? Because you're used to it. Same with bind and match
Man I'm so glad I took a functional programming course last semester at my Uni, monads are so awesome!
Man, I wish they taught FP in my uni when I studied 😂
Looks nice and fancy afterwards, but by wrapping everything in static extension methods I would argue it is harder to read, debug and unit test. The original way was clean, simple, effective and easily tested.
Tell me you've never done functional programming before
@@MilanJovanovicTech dude, I'd love to follow your content, but this response is disappointing. It's just a bit childish and unconstructive.
@@Cruize91 Why is it rude? The main critics of this video (and any video I create about functional programming) are people who don't do any FP. They see everything through an OOP lens, and can't admit that pure functional code cam indeed be clean, simple, effective, and easily tested.
Oh, and don't judge a person based on one comment... 😅
@@MilanJovanovicTech I didn't judge your person at all. I criticized the response.
Secondly, you're putting words in my mouth. I didn't say 'rude'.
You have two commenters here who are genuinely trying to interact with you and your content.
The first comment was simply a voice of not understanding. You could've handled that so much better by replying with a friendly stock answer, if this is something that occurs so often.
Secondly, I'm simply stating that I disagree with this kind of response, because aren't we here to learn? A guy comments with some genuine concerns, so it's a chance to educate nicely.
I'm not saying this to scold you, I'm just saying that I like your content, and would like to keep following it, but this kind of behavior is gonna turn me away.
Amazing content! You’re always surprise me with very useful content, thanks!
Happy to hear that!
Man your contents are just amazing.keep it up. good luck!
Thanks, will do!
I would prefer to have the old version for 3 reasons: easy to read, easy to debug, looks native.
Only because most people are not used to FP style code. Otherwise, this is also easy to read, and if anything it's even easier to read, since it follows a very natural flow from step to the other.
@@michbushi Try a little rxjs or Reactive Extensions for .NET then come back to this. It's a mindset shift.
Milan Nice Explanation.
One Suggestion , like you're using already created project for Video.
I think it's hard for viewers to understand when you start coding in already created classes etc.
I think it will be good if you demo it in sample small project.
it will be easy to understand. :)
I don't really have the luxury of going over a project from scratch in every video, since they would be too long and people would lose interest. :/
You've created a Result monad. This comes out of the box in most functional languages. It's such a round-about way of doing it in C# that I usually just don't bother writing all of the "extra code to make it more F#'y".
The Result class makes it possible to make a distinction between an expected (validation / domain) error and an actual unexpected exception (e.g. the database is down or we accidently introduced a bug).
I find it worthwhile going through the trouble, since I'm a fan of FP in C#
@@MilanJovanovicTech fair enough.
Thank you for providing such a great content. I found it very useful and helpful 😊
You're welcome!
Great job Milan. Would be great to see a GraphQL video soon
That's on my learning list, since I didn't work with GraphQL before
How do you not have to write authentication scheme on the controller function? Which one is picked as default? If you only have one active, will it be picked as default?
You can place it on controller level
@@MilanJovanovicTech ah, I missed that!
Super content. Thank you!, Milan
You're welcome!
Please make a video on how to implement identity server with role based auth + how to check the role conditionally in a common/shared method
I was thinking of covering some other IDP actually
Can you make a series on functional programming important concepts Milan ?
I just recorded one more about ROP, releasing in a few weeks
@@MilanJovanovicTech Thanks very much for all your effort
Thanks for this content. Any suggestions on how to learn FP please tell us.
Practice FP daily :)
@@MilanJovanovicTech any articles?
Great idea for refactoring! Could you make a introduction video on monads and discriminated unions in C#?
I think Zoran Horvat is much more adept to talk about those subjects, but I'll give it a try why not 😁
I prefer fluent way as well, but performance has to be taken into consideration if it’s a priority 🚀
Always appreciate an FP fan :)
On a side note, what is the benefit of using that pattern, is it simply to adhere to the "Clean architecture setup"?
Performance wise, how is this better/worse than simply having some interface IService with method DoSomething(), and then injecting that service into the application?
Of course I think the way you do it in this video is nice, but it also requires a little more boilerplate to get running.
And lastly, thanks for the videos, I love them :)
Performance wise it's slightly worse, because of more allocations and method calls.
But FP is very nice to work with, although it takes some time getting used to.
The end result is the same. I'm just trying to raise awareness about FP.
We receive REST payload object to create another request inside controller and process it somewhere else? Well... It's far better to encapsulate logic in minimal API or FastEndpoints than using another layer.
One is external (public) to the API, the other is internal to your application
@MilanJovanovicTech, that is a super clean and easy to read design, thanks for your great ideas and inspirations. Don't pay to much attention to ones who say it's complex and unreadable. Even for average mind as mine it is quite straightforward (even given I have started to learn c# dotnet a few month ago).
I could only think of better name for Bind, like Execute, but probably you have some convention as I see you always do in dotnet)
I have a few more videos on ROP, so if you're up for it check them out:
- th-cam.com/video/dDasAmowFts/w-d-xo.html
- th-cam.com/video/zuy2j8vxgYc/w-d-xo.html
- th-cam.com/video/vGkgsduwnc4/w-d-xo.html
Hi Milan. As much as I love the extension methods and the flexibility and clean code practices they bring to the codebase I'm skeptical of their impact on performance/memory. Static classes gets instantiated as soon as the application starts and they live in heap memory throughout application's lifetime. I'm not sure if using too many static classes/methods is going to have a significant impact on application's performance as it will cause frequent GC triggers? Correct me if I'm wrong please. Thankyou.
Static classes get never instantiated, since no objects are created thereby.
We have bigger problems to solve than a few extra allocations/method calls
@@krccmsitp2884 But they live in heap memory forever. Sorry used the incorrect term 'instantiation'. One can say intilization I guess?
@@MilanJovanovicTech That's where Nick and you might differ I guess. But I agree most clients are either unaware of or simply don't care about memory efficiency and performance. They just want working solution with minimal overhead.
@@vivekkaushik9508 there's no heap allocation involved with static classes since there's no memory consumption necessary.
Love this with the ErrorOr library, have something very similar in place currently. In my opinion throwing Exceptions for errors which are expected (we know about) is logically incoherent since they are not exceptional in nature. What's your take on Errors vs Exceptions?
I'm on the Errors side
I think that instead of creating additional abstraction with those Bind and Match methods it could be simplified just by not going back to the controller with IsFailure, Value, Errors properties, instead returning the actual result (value) from the query and just a Task from command, and if there is a validation/business logic error - just thrown an exception with proper context and handle it to returned desired code & message
in this way in the controller it would be just a 2 liner:
public async Task UpdateMember(Guid id, [FromBody] UpdateMemberRequest request, CancellationToken cancellationToken)
{
await Sender.Send(request);
return NoContent();
}
Why would I have a Result object then - which is expressive - and obscure it with throwing an exception somewhere in the code?
@@MilanJovanovicTech let's assume that you send a command that violates some business rules that are checked in the domain layer, in this approach instead of throwing an exception, you have to pass information up from the domain layer about whether it failed or not, thus coupling domain, application and presentation layers by branching the 'IsFailure' property
Hi Milan, nice and efficient work, thanks for sharing it! I see you are using CancellationToken parameter in your Post Methods. I heard that we should not be using CancellationToken in our Post Methods because of in case of cancel there may be an inconsistency in the data as a result of the process being interrupted. What do you think about this?
If you have one DB transaction it'll be fine
Nice video! Btw, what's the name of you VS theme?
ReSharper's dark theme
that is selectmany right?
What is?
@@MilanJovanovicTech bind i think is the same as .SelectMany, also i think if you name it that insted of bind you can use the "from" "in" keywords
Why use fp over oop?
Because FP is simply awesome
@@MilanJovanovicTech That's not a good reason.
Can you please make some videos on aws cloud formation + azure pipeline using .Net core , Wanted detail video how API triggers lambda through cloud formation.
That's a very specific topic, I'll think about it
I would prefer the first approach as well . Just keep it simple.
That's fair 😁
Hello Milan,
Love the content.
What do you think about LanguageExt library? Have you thought about doing a video on that?
I think it's great, although I didn't use it before
Hvala puno na lekciji! Chao iz Rusije.
Нема на чему! :)
Wow nice method. I would love to see this on minimal API's of dotnet
It would be pretty cool indeed 😁
Hello Millan! Thanks for the great new video! I always share your videos with my friends in Brazil :)
So I have a question, what do you think about adding a mapper from your DTO to Commands/Queries? like AutoMapper.
I think your controller will have less lines of code, is that a good idea?
I prefer keeping it light on the mappers, and just call constructors directly. And if I do use mapper, I use a dynamic one like Maspter
Prefer/try explicit (or implicit) operators to your DTOs to do the mapping. Yes, you have to write the mapping code yourself, but that's a good thing, especially when it comes to static analysis and debugging.
You can try mapperly, it uses source generators to create static mapping. So at any point you can remove it, and keep the code. It also does not hide references behind reflection. Drawback is that it does not support everything that Mapster does, ex. projections.
Good work, every video we enjoy and learn something new & interesting,
but what if think about time complexity, we usually try to make our codes more simple making reops or like this stuff,
but is it suitable way keeping DATA-STRUCTURE/ALGORITHMS/TIME-COMPLEXITY/ in mind....
please guide.
Thanks.
You'll find that such optimizations are negligible for most applications
@@MilanJovanovicTech yup That's also point... Well thanks buddy
I like it. 🙂
Sweet!
Interesting approach, but I wonder why go to all this trouble when you'd be able to achieve arguably a similar result using Minimal APIs. Also you really should have validation in there somewhere...
Spreading awareness about FP :)
@@MilanJovanovicTech fair enough. Could do the same with minimal as well tho 😉
I don’t see how any of the code in a minimal api would be different/simpler at all. You would just be using Results.* instead of ActionResult.
The body of a minimal API and a controller action are identical, they are functions
This is just way too complicated, my goodness.
Where's the complexity though?
Nice thumbnail haha
Glad I'm finally getting better at making them 😁
Does not extendable for other requirements.
FP is much MORE extendable than OOP - just add another function :)
@@MilanJovanovicTech No it is not. Sctricly Typed languages are horrible for the FP!
That looks similar to the oneof lib
It's called a Monad, and it's a concept from FP. That's why it looks similar
Readable and understanding vs thin controller. And you failed I think)
But you show me interesting solution
Why would you say I failed?
I wished before the video that its not MediatR. Not that I don't like it, just hoped for something new
Haha, sorry to disappoint! 🤣
@@MilanJovanovicTech don't worry 😂 it's not your fault they didn't come up with something better yet