-
Notifications
You must be signed in to change notification settings - Fork 1
Security Trimming
Security trimming is the ability of MvcSiteMapProvider to change the visibility of nodes according to an external security scheme. MvcSiteMapProvider does not actually provide security, it simply shows or hides nodes according to an external security context.
The MVC controller actions, pages, or other resources that those nodes represent must be secured using the built-in features of MVC, ASP.NET, and/or IIS otherwise the URLs will still be available even if the nodes that represent them are not visible.
By default, security trimming is disabled. To use it, the first thing you need to do is turn it on.
<appSettings>
<add key="MvcSiteMapProvider_SecurityTrimmingEnabled" value="true"/>
</appSettings>
bool securityTrimmingEnabled = true; // Near the top of the module
This parameter is passed into the constructor of the SiteMapBuilderSet class.
The recommended way to configure security in MVC is to use the AuthorizeAttribute, or a class that inherits AuthorizeAttribute.
The AuthorizeAttribute can be applied to controllers and actions to specify what roles (or users) have access to the resource (controller or action). If you don't specify a user or role on the AuthorizeAttribute, then all logged in users will have access.
It is usually best if you register AuthorizeAttribute in your global filters, which will make all of your pages require a logged on user by default. Then you can decorate only the actions you want unauthenticated users to access with the AllowAnonymousAttribute. If you do it this way, you enable "white list" security. That is, everything is secure except for what you specifically "white list" using the AllowAnonymousAttribute.
public class FilterConfig
{
public static void RegisterGlobalFilters(GlobalFilterCollection filters)
{
// By default, all actions require a logged in user
filters.Add(new AuthorizeAttribute());
}
}
public class HomeController : Controller
{
// All users have access
[AllowAnonymous]
public ActionResult Index()
{
return View();
}
// Only logged in users have access
public ActionResult ViewSomethingPrivate()
{
return View();
}
}
// Only users in the "Manager" role have access to all actions on this controller
[Authorize(Roles = "Manager")]
public class AccountController : Controller
{
public ActionResult Manage()
{
return View();
}
public ActionResult ChangePassword()
{
return View();
}
}
Once you have your security settings in place and have enabled security trimming, MvcSiteMapProvider will automatically hide the nodes the current user doesn't have access to.
Do note, however, that a child node cannot have lower restrictions than a parent node in the hierarchy. This is usually not a problem because we typically want to hide all of the descendants of a node that the user doesn't have access to. However, this is something that MVC allows that MvcSiteMapProvider does not.
MvcSiteMapProvider will automatically take into account classes that inherit AuthorizeAttribute if you need to customize it in some way.
Note: Inheriting AuthorizeAttribute is the recommended way to extend MVC security.
There is just one restriction - your implementation of AuthorizeAttribute must set the filterContext.Result property to null if it succeeds and set it to a non-null value if it fails. If you override the AuthorizeCore() method only, you are covered because the default implementation does this already. However, you must take this logic into account if you override the OnAuthorization() method.
Sometimes we need to be able to interact with ASP.NET forms authentication as well so we can also hide nodes that represent ASP.NET web forms that are not accessible. In this situation, you can use the roles attribute/property of the node to control which roles have access.
To represent an ASP.NET page, we need to use the URL of that page to represent it.
<mvcSiteMapNode title="Manage Users" url="~/ManageUsers.aspx" roles="Manager,Administrator" />
In this example, only users in the Manager and Administrator roles have access to the /ManageUsers.aspx page.
Note that this works similarly for both Windows and Web forms authentication.
Note: The technique described in this section can only be done using an external DI container.
Sometimes we need to be able to be able to use a custom security scheme other than using AuthorizeAttribute or forms authentication. For this situation, there is the IAclModule extension point. You can implement this interface and inject it using an external DI container.
Here is what a custom IAclModule implementation might look like:
public class MyAclModule
: IAclModule
{
#region IAclModule Members
/// <summary>
/// Determines whether node is accessible to user.
/// </summary>
/// <param name="siteMap">The site map.</param>
/// <param name="node">The node.</param>
/// <returns>
/// <c>true</c> if accessible to user; otherwise, <c>false</c>.
/// </returns>
public bool IsAccessibleToUser(ISiteMap siteMap, ISiteMapNode node)
{
try
{
bool userHasAccessToNode = // TODO: Put logic here to determine this
return userHasAccessToNode;
}
finally
{
// If we cannot reach a conclusion, always return true
return true;
}
}
#endregion
}
Keep in mind, we are not providing security, we are providing visibility. So if we get an exception, we should just make the node visible as the security provider is responsible for actually locking down the resource.
Once we have implemented IAclModule, then we just need to alter the DI configuration (module) to include it. Here is what that would look like if using StructureMap.
// Configure Security
this.For<IAclModule>().Use<CompositeAclModule>()
.EnumerableOf<IAclModule>().Contains(x =>
{
x.Type<AuthorizeAttributeAclModule>();
x.Type<XmlRolesAclModule>();
x.Type<MyAclModule>(); // Added our new module
});
Note that the IAclModules are processed from the top to the bottom in the list and the first one that returns false wins. If you don't have any interest in using the AuthorizeAtributeAclModule or XmlRolesAclModule after implementing your own, you can safely remove them.
- [Wiki Home] (.)
- [Release Notes] (https://github.com/maartenba/MvcSiteMapProvider/releases)
Want to contribute? See our [Contributing to MvcSiteMapProvider] (https://github.com/maartenba/MvcSiteMapProvider/blob/master/CONTRIBUTING.md) guide.
- [Upgrading from v3 to v4] (Upgrading-from-v3-to-v4)
- [Routing Basics] (Routing-Basics)
- Configuring MvcSiteMapProvider
- Defining Sitemap Nodes in XML
- Defining Sitemap Nodes using .NET Attributes
- Defining Sitemap Nodes using IDynamicNodeProvider
- HtmlHelper Extensions
- Controlling URL Behavior
- Using Action Filter Attributes
- Sitemaps XML Protocol Endpoint for Search Engines
- Using Custom Attributes on a Node
- Specifying Node Order
- Advanced Node Visibility
- Multiple Navigation Paths to a Single Page
- [Multiple Sitemaps in One Application] (Multiple-Sitemaps-in-One-Application)
- [Security Trimming] (Security-Trimming)
[Version 3.x Documentation] (Version-3.x-Documentation)
Other places around the web have some documentation that is helpful for getting started and finding answers that are not found here.
- [MvcSiteMapProvider 4.0 - A Test Drive] (http://www.shiningtreasures.com/post/2013/08/07/MvcSiteMapProvider-40-a-test-drive)
- [MvcSiteMapProvider 4.0 - SEO Features Tutorial] (http://www.shiningtreasures.com/post/2013/08/10/mvcsitemapprovider-4-seo-features)
- [How to Make MvcSiteMapProvider Remember a User’s Position] (http://www.shiningtreasures.com/post/2013/09/02/how-to-make-mvcsitemapprovider-remember-a-user-position)
- [MvcSiteMapProvider 4.0 - Cache Configuration] (http://www.shiningtreasures.com/post/2013/08/11/mvcsitemapprovider-4-cache-configuration)
- [MvcSiteMapProvider 4.0 - Extending the Cache] (http://www.shiningtreasures.com/post/2013/08/13/mvcsitemapprovider-4-extending-the-cache)
- [MvcSiteMapProvider 4.0 - Unit Testing with the SiteMaps Static Methods] (http://www.shiningtreasures.com/post/2013/08/14/mvcsitemapprovider-4-unit-testing-with-the-sitemaps-static-methods)
- [Debugging an MvcSiteMapProvider Configuration] (http://www.shiningtreasures.com/post/2013/08/21/debugging-an-mvcsitemapprovider-configuration)
- [Converting from C# to Vb MvcSiteMapProvider] (http://www.developerfusion.com/thread/112710/converting-from-c-to-vb-mvcsitemapprovider/)
- [NightOwl888's MvcSiteMapProvider Demos] (https://github.com/NightOwl888?tab=repositories) - Filter for "MvcSiteMapProvider" to see the most relevant.
- [MvcSiteMapProvider Tutorial and Examples] (http://edspencer.me.uk/2011/02/10/mvc-sitemap-provider-tutorial/)
- [MvcSiteMapProvider Tutorial 2 - Breadcrumbs] (http://edspencer.me.uk/2011/09/20/mvc-sitemap-provider-tutorial-2-breadcrumbs/)
- [Getting Started with MvcSiteMapProvider] (http://blog.danstuken.com/2011/04/29/getting-started-with-mvcsitemapprovider/)
- [Inside the MvcSiteMapProvider - Part 1] (http://xharze.blogspot.com/2012/04/inside-mvcsitemapprovider-part-1.html)
- [Inside the MvcSiteMapProvider - Part 2: Dynamic node providers] (http://xharze.blogspot.com/2012/04/inside-mvcsitemapprovider-part-2.html)
- [Inside the MvcSiteMapProvider - Part 3: The ISiteMapVisibilityProvider] (http://xharze.blogspot.com/2012/04/inside-mvcsitemapprovider-part-3.html)
- [Inside the MvcSiteMapProvider - Part 4: The IAclModule] (http://xharze.blogspot.com/2012/04/inside-mvcsitemapprovider-part-4.html)
- [Inside the MvcSiteMapProvider - Part 5: The ISiteMapNodeUrlResolver] (http://xharze.blogspot.com/2012/04/inside-mvcsitemapprovider-part-5.html)
- [Styling MvcSiteMapProvider with CSS] (http://tutsblog.net/styling-mvc-sitemap-provider-with-css/)
- [Using MvcSiteMapProvider with Twitter Bootstrap] (http://codingit.wordpress.com/2013/05/03/using-mvcsitemapprovider-with-twitter-bootstrap/)
- [ASP.NET MVC Menu using Site Map Provider & Bootstrap Navbar] (http://joeylicc.wordpress.com/2013/06/04/asp-net-mvc-menu-using-site-map-provider-bootstrap-navbar/)
- [StackOverflow MvcSiteMapProvider] (http://stackoverflow.com/questions/tagged/mvcsitemapprovider)
- [StackOverflow MvcSiteMap] (http://stackoverflow.com/questions/tagged/mvcsitemap)
- [StackOverflow ASP.NET MVC SiteMap] (http://stackoverflow.com/questions/tagged/asp.net-mvc-sitemap)
- [CodePlex Discussion Forum (no longer maintained)] (http://mvcsitemap.codeplex.com/discussions/topics/general?size=2147483647)
- [Telerik Forum] (http://www.telerik.com/search.aspx?insection=False&start=0&client=telerik_developer_tools&q=MvcSiteMapProvider&sid=1)
- [maartenba's blog] (http://blog.maartenballiauw.be/search.aspx?q=mvcsitemapprovider)
- [NightOwl888's blog] (http://www.shiningtreasures.com/category/MvcSiteMapProvider)