I'm looking for thoughts on how should I use Session in an ASP.NET MVC application? Especially when using masterpages and tryin to just get the data to the masterpage without bypassing the controller. This question started off by me asking a lot of little questions, but then I managed to mould it into a solution which as yet I have not implemented but one which is somewhat workable. Would appreciate any feedback.
My proposed solution aka 'what i am about to implement unless someone says stop!'
I have my model classes inheriting from ModelBase -- which contains the information needed by the masterpage (there is only one view per page) for certain things it displays in the masthead or footer as well as configuration driven settings based upon who is logged in.
My best solution is as follows - shown here for a 'products page':
Assumption: I have at some point already stuck certain data in session - for instance perhaps a
partnerIdwhich came in through a gateway page, or acurrentLoggedInUserEmailproperty or a fully blown object.I have a
ModelBaseclass from which every model - such asProductModelinheritsI have a
MySiteControllerBaseclass (inherits from Controller) - which is subclassed byProductController.In my action method in
ProductControllerI create the model for the product view with'new ProductModel()'. This model class itself knows nothing about session or how to populateModelBase. It essentially doesn't even know aboutModelBase- it just inherits from it. My chained constructor does nothing (because I don't want to pass itSession).I override
View(...)inMySiteControllerBasefor all the overloads that take a model parameter. I check to see if that parameter is of typeModelBaseand if it is I populate the properties such aspartneridandcurrentLoggedInuserEmail. Fortunately because I'm inside a class that inherits fromControllerI have direct access toSessionso i can pull them straight out of there.
This method means that the properties on ModelBase are automatically populated just by me doing 'return View(model)'. However there is an obvious issue if the model for ProductModel needs to access anything defined on ModelBase. It's going to get null because it isn't populated yet.
This issue can be solved by passing in Session to new ProductModel(session) which would in turn pass it up the constructor chain to new ModelBase(session). I really dont like that solution though becasue I like to think of a model as a pretty dumb data structure that shouldn't know about any external data constructs at all. Another solution might be to just wing it, and if i ever find that ProductController needs to consume anything defined in ModelBase that I just create a method MySiteControllerBase.UpdateModelBase(productModel, session) to explicitly populate it inside ProductController. I hope thats clear!
Other questions that come to mind are :
- What about unit testing? Is there any abstraction around Session state in MVC or should I build my own? I did a search in the sourcecode for 'session' and nothing came up!
- How does session tracking work with /REST/FUL/URLS in MVC? Are there any issues ith cookies off that I need to know about?
- Should I think of session differently from how I traditionally have?