e79aa3c0ed
Former-commit-id: a2155e9bd80020e49e72e86c44da02a8ac0e57a4
978 lines
39 KiB
C#
978 lines
39 KiB
C#
//------------------------------------------------------------------------------
|
|
// <copyright file="XmlSiteMapProvider.cs" company="Microsoft">
|
|
// Copyright (c) Microsoft Corporation. All rights reserved.
|
|
// </copyright>
|
|
//------------------------------------------------------------------------------
|
|
|
|
/*
|
|
* XmlSiteMapProvider class definition
|
|
*
|
|
* Copyright (c) 2002 Microsoft Corporation
|
|
*/
|
|
|
|
namespace System.Web {
|
|
|
|
using System;
|
|
using System.Collections;
|
|
using System.Collections.Specialized;
|
|
using System.ComponentModel;
|
|
using System.ComponentModel.Design;
|
|
using System.Configuration;
|
|
using System.Configuration.Provider;
|
|
using System.Diagnostics.CodeAnalysis;
|
|
using System.Globalization;
|
|
using System.IO;
|
|
using System.Resources;
|
|
using System.Security;
|
|
using System.Security.Permissions;
|
|
using System.Web.Compilation;
|
|
using System.Web.Configuration;
|
|
using System.Web.Hosting;
|
|
using System.Web.UI;
|
|
using System.Web.Util;
|
|
using System.Xml;
|
|
|
|
// XmlMapProvider that generates sitemap tree from xml files
|
|
|
|
public class XmlSiteMapProvider : StaticSiteMapProvider, IDisposable {
|
|
|
|
private string _filename;
|
|
private VirtualPath _virtualPath;
|
|
private VirtualPath _normalizedVirtualPath;
|
|
private SiteMapNode _siteMapNode;
|
|
private XmlDocument _document;
|
|
private bool _initialized;
|
|
private FileChangeEventHandler _handler;
|
|
private StringCollection _parentSiteMapFileCollection;
|
|
|
|
private const string _providerAttribute = "provider";
|
|
private const string _siteMapFileAttribute = "siteMapFile";
|
|
private const string _siteMapNodeName = "siteMapNode";
|
|
private const string _xmlSiteMapFileExtension = ".sitemap";
|
|
private const string _resourcePrefix = "$resources:";
|
|
private const int _resourcePrefixLength = 10;
|
|
private const char _resourceKeySeparator = ',';
|
|
private static readonly char[] _seperators = new char[] { ';', ',' };
|
|
|
|
private ArrayList _childProviderList;
|
|
|
|
// table containing mappings from child providers to their root nodes.
|
|
private Hashtable _childProviderTable;
|
|
|
|
|
|
public XmlSiteMapProvider() {
|
|
}
|
|
|
|
private ArrayList ChildProviderList {
|
|
get {
|
|
ArrayList returnList = _childProviderList;
|
|
if (returnList == null) {
|
|
lock (_lock) {
|
|
if (_childProviderList == null) {
|
|
returnList = ArrayList.ReadOnly(new ArrayList(ChildProviderTable.Keys));
|
|
_childProviderList = returnList;
|
|
}
|
|
else {
|
|
returnList = _childProviderList;
|
|
}
|
|
}
|
|
}
|
|
|
|
return returnList;
|
|
}
|
|
}
|
|
|
|
private Hashtable ChildProviderTable {
|
|
get {
|
|
if (_childProviderTable == null) {
|
|
lock (_lock) {
|
|
if (_childProviderTable == null) {
|
|
_childProviderTable = new Hashtable();
|
|
}
|
|
}
|
|
}
|
|
|
|
return _childProviderTable;
|
|
}
|
|
}
|
|
|
|
|
|
public override SiteMapNode RootNode {
|
|
get {
|
|
BuildSiteMap();
|
|
SiteMapNode node = ReturnNodeIfAccessible(_siteMapNode);
|
|
return ApplyModifierIfExists(node);
|
|
}
|
|
}
|
|
|
|
public override SiteMapNode CurrentNode {
|
|
get {
|
|
return ApplyModifierIfExists(base.CurrentNode);
|
|
}
|
|
}
|
|
|
|
public override SiteMapNode GetParentNode(SiteMapNode node) {
|
|
SiteMapNode parentNode = base.GetParentNode(node);
|
|
return ApplyModifierIfExists(parentNode);
|
|
}
|
|
|
|
public override SiteMapNodeCollection GetChildNodes(SiteMapNode node) {
|
|
SiteMapNodeCollection subNodes = base.GetChildNodes(node);
|
|
HttpContext context = HttpContext.Current;
|
|
|
|
// Do nothing if the modifier doesn't apply
|
|
if (context == null || !context.Response.UsePathModifier || subNodes.Count == 0) {
|
|
return subNodes;
|
|
}
|
|
|
|
// Apply the modifier to the children nodes
|
|
SiteMapNodeCollection resultNodes = new SiteMapNodeCollection(subNodes.Count);
|
|
|
|
foreach (SiteMapNode n in subNodes) {
|
|
resultNodes.Add(ApplyModifierIfExists(n));
|
|
}
|
|
|
|
return resultNodes;
|
|
}
|
|
|
|
protected internal override void AddNode(SiteMapNode node, SiteMapNode parentNode) {
|
|
if (node == null) {
|
|
throw new ArgumentNullException("node");
|
|
}
|
|
|
|
if (parentNode == null) {
|
|
throw new ArgumentNullException("parentNode");
|
|
}
|
|
|
|
SiteMapProvider ownerProvider = node.Provider;
|
|
SiteMapProvider parentOwnerProvider = parentNode.Provider;
|
|
|
|
if (ownerProvider != this) {
|
|
throw new ArgumentException(SR.GetString(SR.XmlSiteMapProvider_cannot_add_node, node.ToString()), "node");
|
|
}
|
|
|
|
if (parentOwnerProvider != this) {
|
|
throw new ArgumentException(SR.GetString(SR.XmlSiteMapProvider_cannot_add_node, parentNode.ToString()), "parentNode");
|
|
}
|
|
|
|
lock (_lock) {
|
|
// First remove it from its current location.
|
|
RemoveNode(node);
|
|
AddNodeInternal(node, parentNode, null);
|
|
}
|
|
}
|
|
|
|
private void AddNodeInternal(SiteMapNode node, SiteMapNode parentNode, XmlNode xmlNode) {
|
|
lock (_lock) {
|
|
String url = node.Url;
|
|
String key = node.Key;
|
|
|
|
bool isValidUrl = false;
|
|
|
|
// Only add the node to the url table if it's a static node.
|
|
if (!String.IsNullOrEmpty(url)) {
|
|
if (UrlTable[url] != null) {
|
|
if (xmlNode != null) {
|
|
throw new ConfigurationErrorsException(
|
|
SR.GetString(SR.XmlSiteMapProvider_Multiple_Nodes_With_Identical_Url, url),
|
|
xmlNode);
|
|
}
|
|
else {
|
|
throw new InvalidOperationException(SR.GetString(
|
|
SR.XmlSiteMapProvider_Multiple_Nodes_With_Identical_Url, url));
|
|
}
|
|
}
|
|
|
|
isValidUrl = true;
|
|
}
|
|
|
|
if (KeyTable.Contains(key)) {
|
|
if (xmlNode != null) {
|
|
throw new ConfigurationErrorsException(
|
|
SR.GetString(SR.XmlSiteMapProvider_Multiple_Nodes_With_Identical_Key, key),
|
|
xmlNode);
|
|
}
|
|
else {
|
|
throw new InvalidOperationException(
|
|
SR.GetString(SR.XmlSiteMapProvider_Multiple_Nodes_With_Identical_Key, key));
|
|
}
|
|
}
|
|
|
|
if (isValidUrl) {
|
|
UrlTable[url] = node;
|
|
}
|
|
|
|
KeyTable[key] = node;
|
|
|
|
// Add the new node into parentNode collection
|
|
if (parentNode != null) {
|
|
ParentNodeTable[node] = parentNode;
|
|
|
|
if (ChildNodeCollectionTable[parentNode] == null) {
|
|
ChildNodeCollectionTable[parentNode] = new SiteMapNodeCollection();
|
|
}
|
|
|
|
((SiteMapNodeCollection)ChildNodeCollectionTable[parentNode]).Add(node);
|
|
}
|
|
}
|
|
}
|
|
|
|
protected virtual void AddProvider(string providerName, SiteMapNode parentNode) {
|
|
if (parentNode == null) {
|
|
throw new ArgumentNullException("parentNode");
|
|
}
|
|
|
|
if (parentNode.Provider != this) {
|
|
throw new ArgumentException(SR.GetString(SR.XmlSiteMapProvider_cannot_add_node, parentNode.ToString()), "parentNode");
|
|
}
|
|
|
|
SiteMapNode node = GetNodeFromProvider(providerName);
|
|
AddNodeInternal(node, parentNode, null);
|
|
}
|
|
|
|
|
|
[SuppressMessage("Microsoft.Security", "MSEC1205:DoNotAllowDtdOnXmlTextReader", Justification = "Legacy code that trusts our developer-controlled input.")]
|
|
public override SiteMapNode BuildSiteMap() {
|
|
|
|
SiteMapNode tempNode = _siteMapNode;
|
|
|
|
// If siteMap is already constructed, simply returns it.
|
|
// Child providers will only be updated when the parent providers need to access them.
|
|
if (tempNode != null) {
|
|
return tempNode;
|
|
}
|
|
|
|
XmlDocument document = GetConfigDocument();
|
|
|
|
lock (_lock) {
|
|
if (_siteMapNode != null) {
|
|
return _siteMapNode;
|
|
}
|
|
|
|
Clear();
|
|
|
|
// Need to check if the sitemap file exists before opening it.
|
|
CheckSiteMapFileExists();
|
|
|
|
try {
|
|
using (Stream stream = _normalizedVirtualPath.OpenFile()) {
|
|
XmlReader reader = new XmlTextReader(stream);
|
|
document.Load(reader);
|
|
}
|
|
}
|
|
catch (XmlException e) {
|
|
string sourceFile = _virtualPath.VirtualPathString;
|
|
string physicalDir = _normalizedVirtualPath.MapPathInternal();
|
|
if (physicalDir != null && HttpRuntime.HasPathDiscoveryPermission(physicalDir)) {
|
|
sourceFile = physicalDir;
|
|
}
|
|
|
|
throw new ConfigurationErrorsException(
|
|
SR.GetString(SR.XmlSiteMapProvider_Error_loading_Config_file, _virtualPath, e.Message),
|
|
e, sourceFile, e.LineNumber);
|
|
}
|
|
catch (Exception e) {
|
|
throw new ConfigurationErrorsException(
|
|
SR.GetString(SR.XmlSiteMapProvider_Error_loading_Config_file, _virtualPath, e.Message), e);
|
|
}
|
|
|
|
XmlNode node = null;
|
|
foreach (XmlNode siteMapMode in document.ChildNodes) {
|
|
if (String.Equals(siteMapMode.Name, "siteMap", StringComparison.Ordinal)) {
|
|
node = siteMapMode;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (node == null)
|
|
throw new ConfigurationErrorsException(
|
|
SR.GetString(SR.XmlSiteMapProvider_Top_Element_Must_Be_SiteMap),
|
|
document);
|
|
|
|
bool enableLocalization = false;
|
|
HandlerBase.GetAndRemoveBooleanAttribute(node, "enableLocalization", ref enableLocalization);
|
|
EnableLocalization = enableLocalization;
|
|
|
|
XmlNode topElement = null;
|
|
foreach (XmlNode subNode in node.ChildNodes) {
|
|
if (subNode.NodeType == XmlNodeType.Element) {
|
|
if (!_siteMapNodeName.Equals(subNode.Name)) {
|
|
throw new ConfigurationErrorsException(
|
|
SR.GetString(SR.XmlSiteMapProvider_Only_SiteMapNode_Allowed),
|
|
subNode);
|
|
}
|
|
|
|
if (topElement != null) {
|
|
throw new ConfigurationErrorsException(
|
|
SR.GetString(SR.XmlSiteMapProvider_Only_One_SiteMapNode_Required_At_Top),
|
|
subNode);
|
|
}
|
|
|
|
topElement = subNode;
|
|
}
|
|
}
|
|
|
|
if (topElement == null) {
|
|
throw new ConfigurationErrorsException(
|
|
SR.GetString(SR.XmlSiteMapProvider_Only_One_SiteMapNode_Required_At_Top),
|
|
node);
|
|
}
|
|
|
|
Queue queue = new Queue(50);
|
|
|
|
// The parentnode of the top node does not exist,
|
|
// simply add a null to satisfy the ConvertFromXmlNode condition.
|
|
queue.Enqueue(null);
|
|
queue.Enqueue(topElement);
|
|
_siteMapNode = ConvertFromXmlNode(queue);
|
|
|
|
return _siteMapNode;
|
|
}
|
|
}
|
|
|
|
private void CheckSiteMapFileExists() {
|
|
if (!System.Web.UI.Util.VirtualFileExistsWithAssert(_normalizedVirtualPath)) {
|
|
throw new InvalidOperationException(
|
|
SR.GetString(SR.XmlSiteMapProvider_FileName_does_not_exist, _virtualPath));
|
|
}
|
|
}
|
|
|
|
|
|
protected override void Clear() {
|
|
lock (_lock) {
|
|
ChildProviderTable.Clear();
|
|
_siteMapNode = null;
|
|
_childProviderList = null;
|
|
|
|
base.Clear();
|
|
}
|
|
}
|
|
|
|
// helper method to convert an XmlNode to a SiteMapNode
|
|
private SiteMapNode ConvertFromXmlNode(Queue queue) {
|
|
|
|
SiteMapNode rootNode = null;
|
|
while (true) {
|
|
if (queue.Count == 0) {
|
|
return rootNode;
|
|
}
|
|
|
|
SiteMapNode parentNode = (SiteMapNode)queue.Dequeue();
|
|
XmlNode xmlNode = (XmlNode)queue.Dequeue();
|
|
|
|
SiteMapNode node = null;
|
|
|
|
if (!_siteMapNodeName.Equals(xmlNode.Name)) {
|
|
throw new ConfigurationErrorsException(
|
|
SR.GetString(SR.XmlSiteMapProvider_Only_SiteMapNode_Allowed),
|
|
xmlNode);
|
|
}
|
|
|
|
string providerName = null;
|
|
HandlerBase.GetAndRemoveNonEmptyStringAttribute(xmlNode, _providerAttribute, ref providerName);
|
|
|
|
// If the siteMapNode references another provider
|
|
if (providerName != null) {
|
|
node = GetNodeFromProvider(providerName);
|
|
|
|
// No other attributes or child nodes are allowed on a provider node.
|
|
HandlerBase.CheckForUnrecognizedAttributes(xmlNode);
|
|
HandlerBase.CheckForNonCommentChildNodes(xmlNode);
|
|
}
|
|
else {
|
|
string siteMapFile = null;
|
|
HandlerBase.GetAndRemoveNonEmptyStringAttribute(xmlNode, _siteMapFileAttribute, ref siteMapFile);
|
|
|
|
if (siteMapFile != null) {
|
|
node = GetNodeFromSiteMapFile(xmlNode, VirtualPath.Create(siteMapFile));
|
|
}
|
|
else {
|
|
node = GetNodeFromXmlNode(xmlNode, queue);
|
|
}
|
|
}
|
|
|
|
AddNodeInternal(node, parentNode, xmlNode);
|
|
|
|
if (rootNode == null) {
|
|
rootNode = node;
|
|
}
|
|
}
|
|
}
|
|
|
|
protected virtual void Dispose(bool disposing) {
|
|
if (_handler != null) {
|
|
Debug.Assert(_filename != null);
|
|
HttpRuntime.FileChangesMonitor.StopMonitoringFile(_filename, _handler);
|
|
}
|
|
}
|
|
|
|
public void Dispose() {
|
|
Dispose(true);
|
|
GC.SuppressFinalize(this);
|
|
}
|
|
|
|
private void EnsureChildSiteMapProviderUpToDate(SiteMapProvider childProvider) {
|
|
SiteMapNode oldNode = (SiteMapNode)ChildProviderTable[childProvider];
|
|
|
|
SiteMapNode newNode = childProvider.GetRootNodeCore();
|
|
if (newNode == null) {
|
|
throw new ProviderException(SR.GetString(SR.XmlSiteMapProvider_invalid_sitemapnode_returned, childProvider.Name));
|
|
}
|
|
|
|
// child providers have been updated.
|
|
if (!oldNode.Equals(newNode)) {
|
|
|
|
// If the child provider table has been updated, simply return null.
|
|
// This will happen when the current provider's sitemap file is changed or Clear() is called;
|
|
if (oldNode == null) {
|
|
return;
|
|
}
|
|
|
|
lock (_lock) {
|
|
oldNode = (SiteMapNode)ChildProviderTable[childProvider];
|
|
// If the child provider table has been updated, simply return null. See above.
|
|
if (oldNode == null) {
|
|
return;
|
|
}
|
|
|
|
newNode = childProvider.GetRootNodeCore();
|
|
if (newNode == null) {
|
|
throw new ProviderException(SR.GetString(SR.XmlSiteMapProvider_invalid_sitemapnode_returned, childProvider.Name));
|
|
}
|
|
|
|
if (!oldNode.Equals(newNode)) {
|
|
|
|
// If the current provider does not contain any nodes but one child provider
|
|
// ie. _siteMapNode == oldNode
|
|
// the oldNode needs to be removed from Url table and the new node will be added.
|
|
if (_siteMapNode.Equals(oldNode)) {
|
|
UrlTable.Remove(oldNode.Url);
|
|
KeyTable.Remove(oldNode.Key);
|
|
|
|
UrlTable.Add(newNode.Url, newNode);
|
|
KeyTable.Add(newNode.Key, newNode);
|
|
|
|
_siteMapNode = newNode;
|
|
}
|
|
|
|
// First find the parent node
|
|
SiteMapNode parent = (SiteMapNode)ParentNodeTable[oldNode];
|
|
|
|
// parent is null when the provider does not contain any static nodes, ie.
|
|
// it only contains definition to include one child provider.
|
|
if (parent != null) {
|
|
// Update the child nodes table
|
|
SiteMapNodeCollection list = (SiteMapNodeCollection)ChildNodeCollectionTable[parent];
|
|
|
|
// Add the newNode to where the oldNode is within parent node's collection.
|
|
int index = list.IndexOf(oldNode);
|
|
if (index != -1) {
|
|
list.Remove(oldNode);
|
|
list.Insert(index, newNode);
|
|
}
|
|
else {
|
|
list.Add(newNode);
|
|
}
|
|
|
|
// Update the parent table
|
|
ParentNodeTable[newNode] = parent;
|
|
ParentNodeTable.Remove(oldNode);
|
|
|
|
// Update the Url table
|
|
UrlTable.Remove(oldNode.Url);
|
|
KeyTable.Remove(oldNode.Key);
|
|
|
|
UrlTable.Add(newNode.Url, newNode);
|
|
KeyTable.Add(newNode.Key, newNode);
|
|
}
|
|
else {
|
|
// Notify the parent provider to update its child provider collection.
|
|
XmlSiteMapProvider provider = ParentProvider as XmlSiteMapProvider;
|
|
if (provider != null) {
|
|
provider.EnsureChildSiteMapProviderUpToDate(this);
|
|
}
|
|
}
|
|
|
|
// Update provider nodes;
|
|
ChildProviderTable[childProvider] = newNode;
|
|
_childProviderList = null;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Returns sitemap node; Search recursively in child providers if not found.
|
|
|
|
public override SiteMapNode FindSiteMapNode(string rawUrl) {
|
|
SiteMapNode node = base.FindSiteMapNode(rawUrl);
|
|
|
|
if (node == null) {
|
|
foreach(SiteMapProvider provider in ChildProviderList) {
|
|
// First make sure the child provider is up-to-date.
|
|
EnsureChildSiteMapProviderUpToDate(provider);
|
|
|
|
node = provider.FindSiteMapNode(rawUrl);
|
|
if (node != null) {
|
|
return node;
|
|
}
|
|
}
|
|
}
|
|
|
|
return node;
|
|
}
|
|
|
|
// Returns sitemap node; Search recursively in child providers if not found.
|
|
public override SiteMapNode FindSiteMapNodeFromKey(string key) {
|
|
SiteMapNode node = base.FindSiteMapNodeFromKey(key);
|
|
|
|
if (node == null) {
|
|
foreach (SiteMapProvider provider in ChildProviderList) {
|
|
// First make sure the child provider is up-to-date.
|
|
EnsureChildSiteMapProviderUpToDate(provider);
|
|
|
|
node = provider.FindSiteMapNodeFromKey(key);
|
|
if (node != null) {
|
|
return node;
|
|
}
|
|
}
|
|
}
|
|
|
|
return node;
|
|
}
|
|
|
|
private XmlDocument GetConfigDocument() {
|
|
if (_document != null)
|
|
return _document;
|
|
|
|
if (!_initialized) {
|
|
throw new InvalidOperationException(
|
|
SR.GetString(SR.XmlSiteMapProvider_Not_Initialized));
|
|
}
|
|
|
|
// Do the error checking here
|
|
if (_virtualPath == null) {
|
|
throw new ArgumentException(
|
|
SR.GetString(SR.XmlSiteMapProvider_missing_siteMapFile, _siteMapFileAttribute));
|
|
}
|
|
|
|
if (!_virtualPath.Extension.Equals(_xmlSiteMapFileExtension, StringComparison.OrdinalIgnoreCase)) {
|
|
throw new InvalidOperationException(
|
|
SR.GetString(SR.XmlSiteMapProvider_Invalid_Extension, _virtualPath));
|
|
}
|
|
|
|
_normalizedVirtualPath = _virtualPath.CombineWithAppRoot();
|
|
_normalizedVirtualPath.FailIfNotWithinAppRoot();
|
|
|
|
// Make sure the file exists
|
|
CheckSiteMapFileExists();
|
|
|
|
_parentSiteMapFileCollection = new StringCollection();
|
|
XmlSiteMapProvider xmlParentProvider = ParentProvider as XmlSiteMapProvider;
|
|
if (xmlParentProvider != null && xmlParentProvider._parentSiteMapFileCollection != null) {
|
|
if (xmlParentProvider._parentSiteMapFileCollection.Contains(_normalizedVirtualPath.VirtualPathString)) {
|
|
throw new InvalidOperationException(
|
|
SR.GetString(SR.XmlSiteMapProvider_FileName_already_in_use, _virtualPath));
|
|
}
|
|
|
|
// Copy the sitemapfiles in used from parent provider to current provider.
|
|
foreach (string filename in xmlParentProvider._parentSiteMapFileCollection) {
|
|
_parentSiteMapFileCollection.Add(filename);
|
|
}
|
|
}
|
|
|
|
// Add current sitemap file to the collection
|
|
_parentSiteMapFileCollection.Add(_normalizedVirtualPath.VirtualPathString);
|
|
|
|
_filename = HostingEnvironment.MapPathInternal(_normalizedVirtualPath);
|
|
|
|
if (!String.IsNullOrEmpty(_filename)) {
|
|
_handler = new FileChangeEventHandler(this.OnConfigFileChange);
|
|
HttpRuntime.FileChangesMonitor.StartMonitoringFile(_filename, _handler);
|
|
ResourceKey = (new FileInfo(_filename)).Name;
|
|
}
|
|
|
|
_document = new ConfigXmlDocument();
|
|
|
|
return _document;
|
|
}
|
|
|
|
private SiteMapNode GetNodeFromProvider(string providerName) {
|
|
SiteMapProvider provider = GetProviderFromName(providerName);
|
|
SiteMapNode node = null;
|
|
|
|
// Check infinite recursive sitemap files
|
|
if (provider is XmlSiteMapProvider) {
|
|
XmlSiteMapProvider xmlProvider = (XmlSiteMapProvider)provider;
|
|
|
|
StringCollection parentSiteMapFileCollection = new StringCollection();
|
|
if (_parentSiteMapFileCollection != null) {
|
|
foreach (string filename in _parentSiteMapFileCollection) {
|
|
parentSiteMapFileCollection.Add(filename);
|
|
}
|
|
}
|
|
|
|
// Make sure the provider is initialized before adding to the collection.
|
|
xmlProvider.BuildSiteMap();
|
|
|
|
parentSiteMapFileCollection.Add(_normalizedVirtualPath.VirtualPathString);
|
|
if (parentSiteMapFileCollection.Contains(VirtualPath.GetVirtualPathString(xmlProvider._normalizedVirtualPath))) {
|
|
throw new InvalidOperationException(SR.GetString(SR.XmlSiteMapProvider_FileName_already_in_use, xmlProvider._virtualPath));
|
|
}
|
|
|
|
xmlProvider._parentSiteMapFileCollection = parentSiteMapFileCollection;
|
|
}
|
|
|
|
node = provider.GetRootNodeCore();
|
|
if (node == null) {
|
|
throw new InvalidOperationException(
|
|
SR.GetString(SR.XmlSiteMapProvider_invalid_GetRootNodeCore, ((ProviderBase)provider).Name));
|
|
}
|
|
|
|
ChildProviderTable.Add(provider, node);
|
|
_childProviderList = null;
|
|
|
|
provider.ParentProvider = this;
|
|
|
|
return node;
|
|
}
|
|
|
|
private SiteMapNode GetNodeFromSiteMapFile(XmlNode xmlNode, VirtualPath siteMapFile) {
|
|
|
|
SiteMapNode node = null;
|
|
|
|
// For external sitemap files, its secuity setting is inherited from parent provider
|
|
bool secuityTrimmingEnabled = SecurityTrimmingEnabled;
|
|
HandlerBase.GetAndRemoveBooleanAttribute(xmlNode, _securityTrimmingEnabledAttrName, ref secuityTrimmingEnabled);
|
|
|
|
// No other attributes or non-comment nodes are allowed on a siteMapFile node
|
|
HandlerBase.CheckForUnrecognizedAttributes(xmlNode);
|
|
HandlerBase.CheckForNonCommentChildNodes(xmlNode);
|
|
|
|
XmlSiteMapProvider childProvider = new XmlSiteMapProvider();
|
|
|
|
// siteMapFile was relative to the sitemap file where this xmlnode is defined, make it an application path.
|
|
siteMapFile = _normalizedVirtualPath.Parent.Combine(siteMapFile);
|
|
|
|
childProvider.ParentProvider = this;
|
|
childProvider.Initialize(siteMapFile, secuityTrimmingEnabled);
|
|
childProvider.BuildSiteMap();
|
|
|
|
node = childProvider._siteMapNode;
|
|
|
|
ChildProviderTable.Add(childProvider, node);
|
|
_childProviderList = null;
|
|
|
|
return node;
|
|
}
|
|
|
|
private void HandleResourceAttribute(XmlNode xmlNode, ref NameValueCollection collection,
|
|
string attrName, ref string text, bool allowImplicitResource) {
|
|
if (String.IsNullOrEmpty(text)) {
|
|
return;
|
|
}
|
|
|
|
string resourceKey = null;
|
|
string temp = text.TrimStart(new char[] { ' ' });
|
|
|
|
if (temp != null && temp.Length > _resourcePrefixLength) {
|
|
if (temp.ToLower(CultureInfo.InvariantCulture).StartsWith(_resourcePrefix, StringComparison.Ordinal)) {
|
|
if (!allowImplicitResource) {
|
|
throw new ConfigurationErrorsException(
|
|
SR.GetString(SR.XmlSiteMapProvider_multiple_resource_definition, attrName), xmlNode);
|
|
}
|
|
|
|
resourceKey = temp.Substring(_resourcePrefixLength + 1);
|
|
|
|
if (resourceKey.Length == 0) {
|
|
throw new ConfigurationErrorsException(
|
|
SR.GetString(SR.XmlSiteMapProvider_resourceKey_cannot_be_empty), xmlNode);
|
|
}
|
|
|
|
// Retrieve className from attribute
|
|
string className = null;
|
|
string key = null;
|
|
int index = resourceKey.IndexOf(_resourceKeySeparator);
|
|
if (index == -1) {
|
|
throw new ConfigurationErrorsException(
|
|
SR.GetString(SR.XmlSiteMapProvider_invalid_resource_key, resourceKey), xmlNode);
|
|
}
|
|
|
|
className = resourceKey.Substring(0, index);
|
|
key = resourceKey.Substring(index + 1);
|
|
|
|
// Retrieve resource key and default value from attribute
|
|
int defaultIndex = key.IndexOf(_resourceKeySeparator);
|
|
if (defaultIndex != -1) {
|
|
text = key.Substring(defaultIndex + 1);
|
|
key = key.Substring(0, defaultIndex);
|
|
}
|
|
else {
|
|
text = null;
|
|
}
|
|
|
|
if (collection == null) {
|
|
collection = new NameValueCollection();
|
|
}
|
|
|
|
collection.Add(attrName, className.Trim());
|
|
collection.Add(attrName, key.Trim());
|
|
}
|
|
}
|
|
}
|
|
|
|
private SiteMapNode GetNodeFromXmlNode(XmlNode xmlNode, Queue queue) {
|
|
SiteMapNode node = null;
|
|
// static nodes
|
|
string title = null, url = null, description = null, roles = null, resourceKey = null;
|
|
|
|
// Url attribute is NOT required for a xml node.
|
|
HandlerBase.GetAndRemoveStringAttribute(xmlNode, "url", ref url);
|
|
HandlerBase.GetAndRemoveStringAttribute(xmlNode, "title", ref title);
|
|
HandlerBase.GetAndRemoveStringAttribute(xmlNode, "description", ref description);
|
|
HandlerBase.GetAndRemoveStringAttribute(xmlNode, "roles", ref roles);
|
|
HandlerBase.GetAndRemoveStringAttribute(xmlNode, "resourceKey", ref resourceKey);
|
|
|
|
// Do not add the resourceKey if the resource is not valid.
|
|
if (!String.IsNullOrEmpty(resourceKey) &&
|
|
!ValidateResource(ResourceKey, resourceKey + ".title")) {
|
|
resourceKey = null;
|
|
}
|
|
|
|
HandlerBase.CheckForbiddenAttribute(xmlNode, _securityTrimmingEnabledAttrName);
|
|
|
|
NameValueCollection resourceKeyCollection = null;
|
|
bool allowImplicitResourceAttribute = String.IsNullOrEmpty(resourceKey);
|
|
HandleResourceAttribute(xmlNode, ref resourceKeyCollection,
|
|
"title", ref title, allowImplicitResourceAttribute);
|
|
HandleResourceAttribute(xmlNode, ref resourceKeyCollection,
|
|
"description", ref description, allowImplicitResourceAttribute);
|
|
|
|
ArrayList roleList = new ArrayList();
|
|
if (roles != null) {
|
|
int foundIndex = roles.IndexOf('?');
|
|
if (foundIndex != -1) {
|
|
throw new ConfigurationErrorsException(
|
|
SR.GetString(SR.Auth_rule_names_cant_contain_char,
|
|
roles[foundIndex].ToString(CultureInfo.InvariantCulture)), xmlNode);
|
|
}
|
|
|
|
foreach (string role in roles.Split(_seperators)) {
|
|
string trimmedRole = role.Trim();
|
|
if (trimmedRole.Length > 0) {
|
|
roleList.Add(trimmedRole);
|
|
}
|
|
}
|
|
}
|
|
roleList = ArrayList.ReadOnly(roleList);
|
|
|
|
String key = null;
|
|
|
|
// Make urls absolute.
|
|
if (!String.IsNullOrEmpty(url)) {
|
|
// URL needs to be trimmed. VSWhidbey 411041
|
|
url = url.Trim();
|
|
|
|
if (!UrlPath.IsAbsolutePhysicalPath(url)) {
|
|
if (UrlPath.IsRelativeUrl(url)) {
|
|
url = UrlPath.Combine(HttpRuntime.AppDomainAppVirtualPathString, url);
|
|
}
|
|
}
|
|
|
|
// VSWhidbey 418056, Reject any suspicious or mal-formed Urls.
|
|
string decodedUrl = HttpUtility.UrlDecode(url);
|
|
if (!String.Equals(url, decodedUrl, StringComparison.Ordinal)) {
|
|
throw new ConfigurationErrorsException(
|
|
SR.GetString(SR.Property_Had_Malformed_Url, "url", url), xmlNode);
|
|
}
|
|
|
|
key = url.ToLowerInvariant();
|
|
}
|
|
else {
|
|
key = Guid.NewGuid().ToString();
|
|
}
|
|
|
|
// attribute collection does not contain pre-defined properties like title, url, etc.
|
|
ReadOnlyNameValueCollection attributeCollection = new ReadOnlyNameValueCollection();
|
|
attributeCollection.SetReadOnly(false);
|
|
foreach (XmlAttribute attribute in xmlNode.Attributes) {
|
|
string value = attribute.Value;
|
|
HandleResourceAttribute(xmlNode, ref resourceKeyCollection, attribute.Name, ref value, allowImplicitResourceAttribute);
|
|
attributeCollection[attribute.Name] = value;
|
|
}
|
|
attributeCollection.SetReadOnly(true);
|
|
|
|
node = new SiteMapNode(this, key, url, title, description, roleList, attributeCollection, resourceKeyCollection, resourceKey);
|
|
node.ReadOnly = true;
|
|
|
|
foreach (XmlNode subNode in xmlNode.ChildNodes) {
|
|
if (subNode.NodeType != XmlNodeType.Element)
|
|
continue;
|
|
|
|
queue.Enqueue(node);
|
|
queue.Enqueue(subNode);
|
|
}
|
|
|
|
return node;
|
|
}
|
|
|
|
private SiteMapProvider GetProviderFromName(string providerName) {
|
|
Debug.Assert(providerName != null);
|
|
|
|
SiteMapProvider provider = SiteMap.Providers[providerName];
|
|
if (provider == null) {
|
|
throw new ProviderException(SR.GetString(SR.Provider_Not_Found, providerName));
|
|
}
|
|
|
|
return provider;
|
|
}
|
|
|
|
protected internal override SiteMapNode GetRootNodeCore() {
|
|
BuildSiteMap();
|
|
return _siteMapNode;
|
|
}
|
|
|
|
|
|
public override void Initialize(string name, NameValueCollection attributes) {
|
|
if (_initialized) {
|
|
throw new InvalidOperationException(
|
|
SR.GetString(SR.XmlSiteMapProvider_Cannot_Be_Inited_Twice));
|
|
}
|
|
|
|
if (attributes != null) {
|
|
if (string.IsNullOrEmpty(attributes["description"])) {
|
|
attributes.Remove("description");
|
|
attributes.Add("description", SR.GetString(SR.XmlSiteMapProvider_Description));
|
|
}
|
|
|
|
string siteMapFile = null;
|
|
ProviderUtil.GetAndRemoveStringAttribute(attributes, _siteMapFileAttribute, name, ref siteMapFile);
|
|
_virtualPath = VirtualPath.CreateAllowNull(siteMapFile);
|
|
}
|
|
|
|
base.Initialize(name, attributes);
|
|
|
|
if (attributes != null) {
|
|
ProviderUtil.CheckUnrecognizedAttributes(attributes, name);
|
|
}
|
|
|
|
_initialized = true;
|
|
}
|
|
|
|
private void Initialize(VirtualPath virtualPath, bool secuityTrimmingEnabled) {
|
|
NameValueCollection coll = new NameValueCollection();
|
|
coll.Add(_siteMapFileAttribute, virtualPath.VirtualPathString);
|
|
coll.Add(_securityTrimmingEnabledAttrName, System.Web.UI.Util.GetStringFromBool(secuityTrimmingEnabled));
|
|
|
|
// Use the siteMapFile virtual path as the provider name
|
|
Initialize(virtualPath.VirtualPathString, coll);
|
|
}
|
|
|
|
private void OnConfigFileChange(Object sender, FileChangeEvent e) {
|
|
// Notifiy the parent for the change.
|
|
XmlSiteMapProvider parentProvider = ParentProvider as XmlSiteMapProvider;
|
|
if (parentProvider != null) {
|
|
parentProvider.OnConfigFileChange(sender, e);
|
|
}
|
|
|
|
Clear();
|
|
}
|
|
|
|
protected internal override void RemoveNode(SiteMapNode node) {
|
|
if (node == null) {
|
|
throw new ArgumentNullException("node");
|
|
}
|
|
|
|
SiteMapProvider ownerProvider = node.Provider;
|
|
|
|
if (ownerProvider != this) {
|
|
|
|
// Only nodes defined in this provider tree can be removed.
|
|
SiteMapProvider parentProvider = ownerProvider.ParentProvider;
|
|
while (parentProvider != this) {
|
|
if (parentProvider == null) {
|
|
// Cannot remove nodes defined in other providers
|
|
throw new InvalidOperationException(
|
|
SR.GetString(SR.XmlSiteMapProvider_cannot_remove_node, node.ToString(),
|
|
this.Name, ownerProvider.Name));
|
|
}
|
|
|
|
parentProvider = parentProvider.ParentProvider;
|
|
}
|
|
}
|
|
|
|
if (node.Equals(ownerProvider.GetRootNodeCore())) {
|
|
throw new InvalidOperationException(SR.GetString(SR.SiteMapProvider_cannot_remove_root_node));
|
|
}
|
|
|
|
if (ownerProvider != this) {
|
|
// Remove node from the owner provider.
|
|
ownerProvider.RemoveNode(node);
|
|
}
|
|
|
|
base.RemoveNode(node);
|
|
}
|
|
|
|
protected virtual void RemoveProvider(string providerName) {
|
|
if (providerName == null) {
|
|
throw new ArgumentNullException("providerName");
|
|
}
|
|
|
|
lock (_lock) {
|
|
SiteMapProvider provider = GetProviderFromName(providerName);
|
|
SiteMapNode rootNode = (SiteMapNode)ChildProviderTable[provider];
|
|
|
|
if (rootNode == null) {
|
|
throw new InvalidOperationException(SR.GetString(SR.XmlSiteMapProvider_cannot_find_provider, provider.Name, this.Name));
|
|
}
|
|
|
|
provider.ParentProvider = null;
|
|
ChildProviderTable.Remove(provider);
|
|
_childProviderList = null;
|
|
|
|
base.RemoveNode(rootNode);
|
|
}
|
|
}
|
|
|
|
// VSWhidbey: 493981 Helper method to check if the valid resource type exists.
|
|
// Note that this only returns false if the classKey cannot be found, regardless of resourceKey.
|
|
private bool ValidateResource(string classKey, string resourceKey) {
|
|
try {
|
|
HttpContext.GetGlobalResourceObject(classKey, resourceKey);
|
|
}
|
|
catch (MissingManifestResourceException) {
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
// Dev10# 923217 - SiteMapProvider URL Table Invalid Using Cookieless
|
|
// Don't keep the modifier inside the links table. Apply the modifier as approriate on demand
|
|
private static SiteMapNode ApplyModifierIfExists(SiteMapNode node) {
|
|
HttpContext context = HttpContext.Current;
|
|
|
|
// Do nothing if the modifier doesn't apply
|
|
if (node == null || context == null || !context.Response.UsePathModifier) {
|
|
return node;
|
|
}
|
|
|
|
// Set Url with the modifier applied
|
|
SiteMapNode resultNode = node.Clone();
|
|
resultNode.Url = context.Response.ApplyAppPathModifier(node.Url);
|
|
|
|
return resultNode;
|
|
}
|
|
|
|
private class ReadOnlyNameValueCollection : NameValueCollection {
|
|
|
|
public ReadOnlyNameValueCollection() {
|
|
IsReadOnly = true;
|
|
}
|
|
|
|
internal void SetReadOnly(bool isReadonly) {
|
|
IsReadOnly = isReadonly;
|
|
}
|
|
}
|
|
}
|
|
}
|