MVC DDD: Is it OK to use repositories together with services in the controller?

前端 未结 5 500
情歌与酒
情歌与酒 2020-12-31 15:51

most of the time in the service code I would have something like this:

public SomeService : ISomeService
{
    ISomeRepository someRepository;
    public Do(         


        
相关标签:
5条回答
  • 2020-12-31 16:19

    My own rough practices for DDD/MVC:

    • controllers are application-specific, hence they should only contain application-specific methods, and call Services methods.
    • all public Service methods are usually atomic transactions or queries
    • only Services instantiate & call Repositories
    • my Domain defines an IContextFactory and an IContext (massive leaky abstraction as IContext members are IDBSet)
    • each application has a sort-of Composition Root, which is mainly instantiating a Context Factory to pass to the Services (you could use DI container but not a big deal)

    This forces me to keep my business code, and data-access out of my controllers. I find it a good discipline, given how loose I am when I don't follow the above!

    0 讨论(0)
  • 2020-12-31 16:33

    You lose the ability to have business logic in between.

    I disagree with this one.

    If business logic is where it should be - in domain model, then calling repo in controller (or better - use model binder for that) to get aggregate root and call method on it seems perfectly fine to me.

    Application services should be used when there's too much technical details involved what would mess up controllers.


    I've seen several people mention using model binders to call into a repo lately. Where is this crazy idea coming from?

    I believe we are talking about 2 different things here. I suspect that Your 'model binder' means using model simultaneously as a view model too and binding changed values from UI directly right back to it (which is not a bad thing per se and in some cases I would go that road).

    My 'model binder' is a class that implements 'IModelBinder', that takes repository in constructor (which is injected and therefore - can be extended if we need caching with some basic composition) and uses it before action is called to retrieve aggregate root and replace int id or Guid id or string slug or whatever action argument with real domain object. Combining that with input view model argument lets us to write less code. Something like this:

    public ActionResult ChangeCustomerAddress
     (Customer c, ChangeCustomerAddressInput inp){
      c.ChangeCustomerAddress(inp.NewAddress);
      return RedirectToAction("Details", new{inp.Id});
    }
    

    In my actual code it's a bit more complex cause it includes ModelState validation and some exception handling that might be thrown from inside of domain model (extracted into Controller extension method for reuse). But not much more. So far - longest controller action is ~10 lines long.

    You can see working implementation (quite sophisticated and (for me) unnecessary complex) here.

    Are you just doing CRUD apps with Linq To Sql or trying something with real domain logic?

    As You can (hopefully) see, this kind of approach actually almost forces us to move towards task based app instead of CRUD based one.

    By doing all data access in your service layer and using IOC you can gain lots of benefits of AOP like invisible caching, transaction management, and easy composition of components that I can't imagine you get with model binders.

    ...and having new abstraction layer that invites us to mix infrastructure with domain logic and lose isolation of domain model.

    Please enlighten me.

    I'm not sure if I did. I don't think that I'm enlightened myself. :)


    Here is my current model binder base class. Here's one of controller actions from my current project. And here's "lack" of business logic.

    0 讨论(0)
  • 2020-12-31 16:37

    Even with "rich domain model" you will still need a domain service for handling business logic which involves several entities. I have also never seen CRUD without some business logic, but in simple sample code. I'd always like to go Martin's route to keep my code straightforward.

    0 讨论(0)
  • 2020-12-31 16:41

    If you use repositories in your controllers, you are going straight from the Data Layer to the Presentation Layer. You lose the ability to have business logic in between.

    Now, if you say you will only use Services when you need business logic, and use Repositories everywhere else, your code becomes a nightmare. The Presentation Layer is now calling both the Business and Data Layer, and you don't have a nice separation of concerns.

    I would always go this route: Repositories -> Services -> UI. As soon as you don't think you need a business layer, the requirements change, and you will have to rewrite EVERYTHING.

    0 讨论(0)
  • 2020-12-31 16:44

    Here is the thing.

    "Business Logic" should reside in your entities and value objects.

    Repositories deal with AggregateRoots only. So, using your repositories directly in your Controllers kinda feels like you are treating that action as your "service". Also, since your AggregateRoot may only refer to other ARs by its ID, you may have to call one more than one repo. It really gets nasty very quickly.

    If you are going to have services, make sure you expose POCOs and not the actual AggregateRoot and its members.

    Your repo shouldn't have to do any operations other than creating, retrieving, updating and deleting stuff. You may have some customized retrieve based on specific conditions, but that's about it. Therefore, having a method in your repo that matches one in your service... code smell right there.

    Your service are API oriented. Think about this... if you were to pack that service in a .dll for me to use, how would you create your methods in a way that is easy for me to know what your service can do? Service.Update(object) doesn't make much sense.

    And I haven't even talked about CQRS... where things get even more interesting.

    Your Web Api is just a CLIENT of your Service. Your Service can be used by another service, right? So, think about it. You most likely will need a Service to encapsulate operations on AggregateRoots, usually by creating them, or retrieving them from a repo, do something about it, then returning a result. Usually.

    Makes sense?

    0 讨论(0)
提交回复
热议问题