1963 lines
99 KiB
C#
Raw Normal View History

//------------------------------------------------------------------------------
// <copyright file="ConfigurationElement.cs" company="Microsoft">
// Copyright (c) Microsoft Corporation. All rights reserved.
// </copyright>
//------------------------------------------------------------------------------
using System;
using System.Configuration.Internal;
using System.Collections;
using System.Collections.Specialized;
using System.Collections.Generic;
using System.IO;
using System.Reflection;
using System.Security.Permissions;
using System.Xml;
using System.Globalization;
using System.ComponentModel;
using System.Security;
using System.Text;
namespace System.Configuration {
//
// Methods that are called by the configuration system, and must be overridable
// by derived classes that wish to implement their own serialization/deserialization
// IsModified()
// ResetModified()
// Reset(ConfigurationElement parentSection, object context)
// DeserializeSection(object context, XmlNode xmlNode)
// SerializeSection(ConfigurationElement parentSection, object context, string name)
//
public abstract class ConfigurationElement {
private const string LockAttributesKey = "lockAttributes";
private const string LockAllAttributesExceptKey = "lockAllAttributesExcept";
private const string LockElementsKey = "lockElements";
private const string LockAll = "*";
private const string LockAllElementsExceptKey = "lockAllElementsExcept";
private const string LockItemKey = "lockItem";
internal const string DefaultCollectionPropertyName = "";
private static string[] s_lockAttributeNames = new string[] {
LockAttributesKey,
LockAllAttributesExceptKey,
LockElementsKey,
LockAllElementsExceptKey,
LockItemKey,
};
private static Hashtable s_propertyBags = new Hashtable();
private static volatile Dictionary<Type,ConfigurationValidatorBase> s_perTypeValidators;
internal static readonly Object s_nullPropertyValue = new Object();
private static ConfigurationElementProperty s_ElementProperty =
new ConfigurationElementProperty(new DefaultValidator());
private bool _bDataToWrite;
private bool _bModified;
private bool _bReadOnly;
private bool _bElementPresent; // Set to false if any part of the element is not inherited
private bool _bInited;
internal ConfigurationLockCollection _lockedAttributesList;
internal ConfigurationLockCollection _lockedAllExceptAttributesList;
internal ConfigurationLockCollection _lockedElementsList;
internal ConfigurationLockCollection _lockedAllExceptElementsList;
private readonly ConfigurationValues _values;
private string _elementTagName;
private volatile ElementInformation _evaluationElement;
private ConfigurationElementProperty _elementProperty = s_ElementProperty;
internal ConfigurationValueFlags _fItemLocked;
internal ContextInformation _evalContext;
internal BaseConfigurationRecord _configRecord;
internal bool DataToWriteInternal {
get {
return _bDataToWrite;
}
set {
_bDataToWrite = value;
}
}
internal ConfigurationElement CreateElement(Type type) {
// We use this.GetType() as the calling type since all code paths which lead to
// CreateElement are protected methods, so inputs are provided by somebody in
// the current type hierarchy. Since we expect that the most subclassed type
// will be the most restricted security-wise, we'll use it as the calling type.
ConfigurationElement element = (ConfigurationElement)TypeUtil.CreateInstanceRestricted(callingType: GetType(), targetType: type);
element.CallInit();
return element;
}
protected ConfigurationElement() {
_values = new ConfigurationValues();
// Set the per-type validator ( this will actually have an effect only for an attributed model elements )
// Note that in the case where the property bag fot this.GetType() has not yet been created
// the validator for this instance will get applied in ApplyValidatorsRecursive ( see this.get_Properties )
ApplyValidator(this);
}
// Give elements that are added to a collection an opportunity to
//
protected internal virtual void Init() {
// If Init is called by the derived class, we may be able
// to set _bInited to true if the derived class properly
// calls Init on its base.
_bInited = true;
}
internal void CallInit() {
// Ensure Init is called just once
if (!_bInited) {
Init();
_bInited = true;
}
}
internal bool ElementPresent {
get {
return _bElementPresent;
}
set {
_bElementPresent = value;
}
}
internal string ElementTagName {
get {
return _elementTagName;
}
}
internal ConfigurationLockCollection LockedAttributesList {
get {
return _lockedAttributesList;
}
}
internal ConfigurationLockCollection LockedAllExceptAttributesList {
get {
return _lockedAllExceptAttributesList;
}
}
internal ConfigurationValueFlags ItemLocked {
get {
return _fItemLocked;
}
}
public ConfigurationLockCollection LockAttributes {
get {
if (_lockedAttributesList == null) {
_lockedAttributesList = new ConfigurationLockCollection(this, ConfigurationLockCollectionType.LockedAttributes);
}
return _lockedAttributesList;
}
}
internal void MergeLocks(ConfigurationElement source) {
if (source != null) {
_fItemLocked = ((source._fItemLocked & ConfigurationValueFlags.Locked) != 0) ?
(ConfigurationValueFlags.Inherited | source._fItemLocked) : _fItemLocked;
if (source._lockedAttributesList != null) {
if (_lockedAttributesList == null) {
_lockedAttributesList = new ConfigurationLockCollection(this, ConfigurationLockCollectionType.LockedAttributes);
}
foreach (string key in source._lockedAttributesList)
_lockedAttributesList.Add(key, ConfigurationValueFlags.Inherited); // Mark entry as from the parent - read only
}
if (source._lockedAllExceptAttributesList != null) {
if (_lockedAllExceptAttributesList == null) {
_lockedAllExceptAttributesList = new ConfigurationLockCollection(this, ConfigurationLockCollectionType.LockedExceptionList, String.Empty, source._lockedAllExceptAttributesList);
}
StringCollection intersectionCollection = IntersectLockCollections(_lockedAllExceptAttributesList, source._lockedAllExceptAttributesList);
_lockedAllExceptAttributesList.ClearInternal(false);
foreach (string key in intersectionCollection) {
_lockedAllExceptAttributesList.Add(key, ConfigurationValueFlags.Default);
}
}
if (source._lockedElementsList != null) {
if (_lockedElementsList == null) {
_lockedElementsList = new ConfigurationLockCollection(this, ConfigurationLockCollectionType.LockedElements);
}
ConfigurationElementCollection collection = null;
if (Properties.DefaultCollectionProperty != null) // this is not a collection but it may contain a default collection
{
collection = this[Properties.DefaultCollectionProperty] as ConfigurationElementCollection;
if (collection != null) {
collection.internalElementTagName = source.ElementTagName; // Default collections don't know there tag name
if (collection._lockedElementsList == null) {
collection._lockedElementsList = _lockedElementsList; //point to the same instance of the collection from parent
}
}
}
foreach (string key in source._lockedElementsList) {
_lockedElementsList.Add(key, ConfigurationValueFlags.Inherited); // Mark entry as from the parent - read only
if (collection != null) {
collection._lockedElementsList.Add(key, ConfigurationValueFlags.Inherited); // add the local copy
}
}
}
if (source._lockedAllExceptElementsList != null) {
if (_lockedAllExceptElementsList == null || _lockedAllExceptElementsList.Count == 0) {
_lockedAllExceptElementsList = new ConfigurationLockCollection(this, ConfigurationLockCollectionType.LockedElementsExceptionList, source._elementTagName, source._lockedAllExceptElementsList);
}
StringCollection intersectionCollection = IntersectLockCollections(_lockedAllExceptElementsList, source._lockedAllExceptElementsList);
ConfigurationElementCollection collection = null;
if (Properties.DefaultCollectionProperty != null) { // this is not a collection but it may contain a default collection
collection = this[Properties.DefaultCollectionProperty] as ConfigurationElementCollection;
if (collection != null && collection._lockedAllExceptElementsList == null) {
// point default collection to the parent collection
collection._lockedAllExceptElementsList = _lockedAllExceptElementsList;
}
}
_lockedAllExceptElementsList.ClearInternal(false);
foreach (string key in intersectionCollection) {
if (!_lockedAllExceptElementsList.Contains(key) || key == ElementTagName)
_lockedAllExceptElementsList.Add(key, ConfigurationValueFlags.Default); // add the local copy
}
if (_lockedAllExceptElementsList.HasParentElements) {
foreach (ConfigurationProperty prop in Properties) {
if ((!_lockedAllExceptElementsList.Contains(prop.Name)) &&
prop.IsConfigurationElementType) {
((ConfigurationElement)this[prop]).SetLocked();
}
}
}
}
}
}
internal void HandleLockedAttributes(ConfigurationElement source) {
// if there are locked attributes on this collection element
if (source != null) {
if (source._lockedAttributesList != null || source._lockedAllExceptAttributesList != null) {
// enumerate the possible locked properties
foreach (PropertyInformation propInfo in source.ElementInformation.Properties) {
if ((source._lockedAttributesList != null && (source._lockedAttributesList.Contains(propInfo.Name) ||
source._lockedAttributesList.Contains(LockAll))) ||
(source._lockedAllExceptAttributesList != null && !source._lockedAllExceptAttributesList.Contains(propInfo.Name))
) {
// if the attribute has been locked in the source then check to see
// if the local config is trying to override it
if (propInfo.Name != LockAttributesKey && propInfo.Name != LockAllAttributesExceptKey) {
if (ElementInformation.Properties[propInfo.Name] == null) { // locked items are not defined
ConfigurationPropertyCollection props = Properties; // so create the property based in the source item
ConfigurationProperty prop = (ConfigurationProperty)source.Properties[propInfo.Name];
props.Add(prop); // Add the property information to the property bag
_evaluationElement = null; // flush the cached element data
// Add the data from the source element but mark it as in herited
// This must use setvalue in order to set the lock and inherited flags
ConfigurationValueFlags flags = ConfigurationValueFlags.Inherited | ConfigurationValueFlags.Locked;
_values.SetValue(propInfo.Name, propInfo.Value, flags, source.PropertyInfoInternal(propInfo.Name));
}
else { // don't error when optional attibute are not defined yet
if (ElementInformation.Properties[propInfo.Name].ValueOrigin == PropertyValueOrigin.SetHere) {
// Don't allow the override
throw new ConfigurationErrorsException(SR.GetString(SR.Config_base_attribute_locked, propInfo.Name));
}
// They did not override so we need to make sure the value comes from the locked one
ElementInformation.Properties[propInfo.Name].Value = propInfo.Value;
}
}
}
}
}
}
}
// AssociateContext
//
// Associate a context with this element
//
internal virtual void AssociateContext(BaseConfigurationRecord configRecord) {
_configRecord = configRecord;
Values.AssociateContext(configRecord);
}
public /*protected internal virtual*/ ConfigurationLockCollection LockAllAttributesExcept {
get {
if (_lockedAllExceptAttributesList == null) {
_lockedAllExceptAttributesList = new ConfigurationLockCollection(this, ConfigurationLockCollectionType.LockedExceptionList, _elementTagName);
}
return _lockedAllExceptAttributesList;
}
}
public ConfigurationLockCollection LockElements {
get {
if (_lockedElementsList == null) {
_lockedElementsList = new ConfigurationLockCollection(this, ConfigurationLockCollectionType.LockedElements);
}
return _lockedElementsList;
}
}
public ConfigurationLockCollection LockAllElementsExcept {
get {
if (_lockedAllExceptElementsList == null) {
_lockedAllExceptElementsList = new ConfigurationLockCollection(this, ConfigurationLockCollectionType.LockedElementsExceptionList, _elementTagName);
}
return _lockedAllExceptElementsList;
}
}
public bool LockItem {
get {
return ((_fItemLocked & ConfigurationValueFlags.Locked) != 0);
}
set {
if ((_fItemLocked & ConfigurationValueFlags.Inherited) == 0) {
_fItemLocked = (value == true) ? ConfigurationValueFlags.Locked : ConfigurationValueFlags.Default;
_fItemLocked |= ConfigurationValueFlags.Modified;
}
else {
throw new ConfigurationErrorsException(SR.GetString(SR.Config_base_attribute_locked, LockItemKey));
}
}
}
protected internal virtual bool IsModified() {
if (_bModified) {
return true;
}
if (_lockedAttributesList != null && _lockedAttributesList.IsModified) {
return true;
}
if (_lockedAllExceptAttributesList != null && _lockedAllExceptAttributesList.IsModified) {
return true;
}
if (_lockedElementsList != null && _lockedElementsList.IsModified) {
return true;
}
if (_lockedAllExceptElementsList != null && _lockedAllExceptElementsList.IsModified) {
return true;
}
if ((_fItemLocked & ConfigurationValueFlags.Modified) != 0) {
return true;
}
foreach (ConfigurationElement elem in _values.ConfigurationElements) {
if (elem.IsModified()) {
return true;
}
}
return false;
}
protected internal virtual void ResetModified() {
_bModified = false;
if (_lockedAttributesList != null) {
_lockedAttributesList.ResetModified();
}
if (_lockedAllExceptAttributesList != null) {
_lockedAllExceptAttributesList.ResetModified();
}
if (_lockedElementsList != null) {
_lockedElementsList.ResetModified();
}
if (_lockedAllExceptElementsList != null) {
_lockedAllExceptElementsList.ResetModified();
}
foreach (ConfigurationElement elem in _values.ConfigurationElements) {
elem.ResetModified();
}
}
public virtual bool IsReadOnly() {
return _bReadOnly;
}
protected internal virtual void SetReadOnly() {
_bReadOnly = true;
foreach (ConfigurationElement elem in _values.ConfigurationElements) {
elem.SetReadOnly();
}
}
internal void SetLocked() {
_fItemLocked = ConfigurationValueFlags.Locked | ConfigurationValueFlags.XMLParentInherited;
foreach (ConfigurationProperty prop in Properties) {
ConfigurationElement elem = this[prop] as ConfigurationElement;
if (elem != null) {
if (elem.GetType() != this.GetType()) {
elem.SetLocked();
}
ConfigurationElementCollection collection = this[prop] as ConfigurationElementCollection;
if (collection != null) {
foreach (object obj in collection) {
ConfigurationElement element = obj as ConfigurationElement;
if (element != null) {
element.SetLocked();
}
}
}
}
}
}
// GetErrorsList
//
// Get the list of Errors for this location and all
// sub locations
//
internal ArrayList GetErrorsList() {
ArrayList errorList = new ArrayList();
ListErrors(errorList);
return errorList;
}
// GetErrors
//
// Get a ConfigurationErrorsException that contains the errors
// for this ConfigurationElement and its children
//
internal ConfigurationErrorsException GetErrors() {
ArrayList errorsList;
errorsList = GetErrorsList();
if (errorsList.Count == 0) {
return null;
}
ConfigurationErrorsException e = new ConfigurationErrorsException(errorsList);
return e;
}
protected virtual void ListErrors(IList errorList) {
// First list errors in this element, then in subelements
foreach (InvalidPropValue invalidValue in _values.InvalidValues) {
errorList.Add(invalidValue.Error);
}
foreach (ConfigurationElement elem in _values.ConfigurationElements) {
elem.ListErrors(errorList);
ConfigurationElementCollection collection = elem as ConfigurationElementCollection;
if (collection != null) {
foreach (ConfigurationElement item in collection) {
item.ListErrors(errorList);
}
}
}
}
protected internal virtual void InitializeDefault() {
}
internal void CheckLockedElement(string elementName, XmlReader reader) {
// have to check if clear was locked!
if(elementName != null) {
if(((_lockedElementsList != null) &&
(_lockedElementsList.DefinedInParent(LockAll) || _lockedElementsList.DefinedInParent(elementName))) ||
((_lockedAllExceptElementsList != null && _lockedAllExceptElementsList.Count != 0) &&
_lockedAllExceptElementsList.HasParentElements &&
!_lockedAllExceptElementsList.DefinedInParent(elementName) ||
(_fItemLocked & ConfigurationValueFlags.Inherited) != 0)
) {
throw new ConfigurationErrorsException(SR.GetString(SR.Config_base_element_locked, elementName), reader);
}
}
}
internal void RemoveAllInheritedLocks() {
if (_lockedAttributesList != null) {
_lockedAttributesList.RemoveInheritedLocks();
}
if (_lockedElementsList != null) {
_lockedElementsList.RemoveInheritedLocks();
}
if (_lockedAllExceptAttributesList != null) {
_lockedAllExceptAttributesList.RemoveInheritedLocks();
}
if (_lockedAllExceptElementsList != null) {
_lockedAllExceptElementsList.RemoveInheritedLocks();
}
}
internal void ResetLockLists(ConfigurationElement parentElement) {
_lockedAttributesList = null;
_lockedAllExceptAttributesList = null;
_lockedElementsList = null;
_lockedAllExceptElementsList = null;
if (parentElement != null) {
_fItemLocked = ((parentElement._fItemLocked & ConfigurationValueFlags.Locked) != 0) ?
(ConfigurationValueFlags.Inherited | parentElement._fItemLocked) :
ConfigurationValueFlags.Default;
if (parentElement._lockedAttributesList != null) {
_lockedAttributesList = new ConfigurationLockCollection(this, ConfigurationLockCollectionType.LockedAttributes);
foreach (string key in parentElement._lockedAttributesList)
_lockedAttributesList.Add(key, ConfigurationValueFlags.Inherited); // Mark entry as from the parent - read only
}
if (parentElement._lockedAllExceptAttributesList != null) {
_lockedAllExceptAttributesList = new ConfigurationLockCollection(this, ConfigurationLockCollectionType.LockedExceptionList, String.Empty, parentElement._lockedAllExceptAttributesList);
}
if (parentElement._lockedElementsList != null) {
_lockedElementsList = new ConfigurationLockCollection(this, ConfigurationLockCollectionType.LockedElements);
ConfigurationElementCollection collection = null;
if (Properties.DefaultCollectionProperty != null) // this is not a collection but it may contain a default collection
{
collection = this[Properties.DefaultCollectionProperty] as ConfigurationElementCollection;
if (collection != null) {
collection.internalElementTagName = parentElement.ElementTagName; // Default collections don't know there tag name
if (collection._lockedElementsList == null) {
collection._lockedElementsList = _lockedElementsList;
}
}
}
foreach (string key in parentElement._lockedElementsList) {
_lockedElementsList.Add(key, ConfigurationValueFlags.Inherited); // Mark entry as from the parent - read only
}
}
if (parentElement._lockedAllExceptElementsList != null) {
_lockedAllExceptElementsList = new ConfigurationLockCollection(this, ConfigurationLockCollectionType.LockedElementsExceptionList, parentElement._elementTagName, parentElement._lockedAllExceptElementsList);
ConfigurationElementCollection collection = null;
if (Properties.DefaultCollectionProperty != null) // this is not a collection but it may contain a default collection
{
collection = this[Properties.DefaultCollectionProperty] as ConfigurationElementCollection;
if (collection != null && collection._lockedAllExceptElementsList == null) {
collection._lockedAllExceptElementsList = _lockedAllExceptElementsList;
}
}
}
}
}
protected internal virtual void Reset(ConfigurationElement parentElement) {
Values.Clear();
ResetLockLists(parentElement);
ConfigurationPropertyCollection props = Properties; // Force the bag to be up to date
_bElementPresent = false;
if (parentElement == null) {
InitializeDefault();
}
else {
bool hasAnyChildElements = false;
ConfigurationPropertyCollection collectionKeys = null;
for (int index = 0; index < parentElement.Values.Count; index++) {
string key = parentElement.Values.GetKey(index);
ConfigurationValue ConfigValue = parentElement.Values.GetConfigValue(index);
object value = (ConfigValue != null) ? ConfigValue.Value : null;
PropertySourceInfo sourceInfo = (ConfigValue != null) ? ConfigValue.SourceInfo : null;
ConfigurationProperty prop = (ConfigurationProperty)parentElement.Properties[key];
if (prop == null || ((collectionKeys != null) && !collectionKeys.Contains(prop.Name))) {
continue;
}
if (prop.IsConfigurationElementType) {
hasAnyChildElements = true;
}
else {
ConfigurationValueFlags flags = ConfigurationValueFlags.Inherited |
(((_lockedAttributesList != null) &&
(_lockedAttributesList.Contains(key) ||
_lockedAttributesList.Contains(LockAll)) ||
(_lockedAllExceptAttributesList != null) &&
!_lockedAllExceptAttributesList.Contains(key)) ?
ConfigurationValueFlags.Locked : ConfigurationValueFlags.Default);
if (value != s_nullPropertyValue) {
// _values[key] = value;
_values.SetValue(key, value, flags, sourceInfo);
}
if (!props.Contains(key)) // this is for optional provider models keys
{
props.Add(prop);
_values.SetValue(key, value, flags, sourceInfo);
}
}
}
if (hasAnyChildElements) {
for (int index = 0; index < parentElement.Values.Count; index++) {
string key = parentElement.Values.GetKey(index);
object value = parentElement.Values[index];
ConfigurationProperty prop = (ConfigurationProperty)parentElement.Properties[key];
if ((prop != null) && prop.IsConfigurationElementType) {
//((ConfigurationElement)value).SerializeToXmlElement(writer, prop.Name);
ConfigurationElement childElement = (ConfigurationElement)this[prop];
childElement.Reset((ConfigurationElement)value);
}
}
}
}
}
public override bool Equals(object compareTo) {
ConfigurationElement compareToElem = compareTo as ConfigurationElement;
if (compareToElem == null ||
(compareTo.GetType() != this.GetType()) ||
((compareToElem != null) && (compareToElem.Properties.Count != this.Properties.Count))) {
return false;
}
foreach (ConfigurationProperty configProperty in this.Properties) {
if (!Object.Equals(Values[configProperty.Name], compareToElem.Values[configProperty.Name])) {
if (!(((Values[configProperty.Name] == null ||
Values[configProperty.Name] == s_nullPropertyValue) &&
Object.Equals(compareToElem.Values[configProperty.Name], configProperty.DefaultValue)) ||
((compareToElem.Values[configProperty.Name] == null ||
compareToElem.Values[configProperty.Name] == s_nullPropertyValue) &&
Object.Equals(Values[configProperty.Name], configProperty.DefaultValue))))
return false;
}
}
return true;
}
public override int GetHashCode() {
int hHashCode = 0;
foreach (ConfigurationProperty configProperty in this.Properties) {
object o = this[configProperty];
if (o != null) {
hHashCode ^= this[configProperty].GetHashCode();
}
}
return hHashCode;
}
protected internal Object this[ConfigurationProperty prop] {
get {
Object o = _values[prop.Name];
if (o == null) {
if (prop.IsConfigurationElementType) {
lock (_values.SyncRoot) {
o = _values[prop.Name];
if (o == null) {
ConfigurationElement childElement = CreateElement(prop.Type);
if (_bReadOnly) {
childElement.SetReadOnly();
}
if (typeof(ConfigurationElementCollection).IsAssignableFrom(prop.Type)) {
ConfigurationElementCollection childElementCollection = childElement as ConfigurationElementCollection;
if (prop.AddElementName != null)
childElementCollection.AddElementName = prop.AddElementName;
if (prop.RemoveElementName != null)
childElementCollection.RemoveElementName = prop.RemoveElementName;
if (prop.ClearElementName != null)
childElementCollection.ClearElementName = prop.ClearElementName;
}
//_values[prop.Name] = childElement;
_values.SetValue(prop.Name, childElement, ConfigurationValueFlags.Inherited, null);
o = childElement;
}
}
}
else {
o = prop.DefaultValue;
}
}
else if (o == s_nullPropertyValue) {
o = null;
}
// If its an invalid value - throw the error now
if (o is InvalidPropValue) {
throw ((InvalidPropValue)o).Error;
}
return o;
}
set {
SetPropertyValue(prop, value,false); // Do not ignore locks!!!
}
}
protected internal Object this[String propertyName] {
get {
ConfigurationProperty prop = Properties[propertyName];
if (prop == null) {
prop = Properties[DefaultCollectionPropertyName];
if (prop.ProvidedName != propertyName) {
return null;
}
}
return this[prop];
}
set {
Debug.Assert(Properties.Contains(propertyName), "Properties.Contains(propertyName)");
SetPropertyValue(Properties[propertyName], value, false);// Do not ignore locks!!!
}
}
// Note: this method is completelly redundant ( the code is duplaicated in ConfigurationProperty( PropertyInfo ) )
// We do not remove the code now to minimize code changes for Whidbey RTM but this method and all calls leading to it should
// be removed post-Whidbey
private static void ApplyInstanceAttributes(object instance) {
Debug.Assert(instance is ConfigurationElement, "instance is ConfigurationElement");
Type type = instance.GetType();
foreach (PropertyInfo propertyInformation in type.GetProperties(BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.Instance)) {
ConfigurationPropertyAttribute attribProperty =
Attribute.GetCustomAttribute(propertyInformation,
typeof(ConfigurationPropertyAttribute)) as ConfigurationPropertyAttribute;
if (attribProperty != null)
{
Type propertyType = propertyInformation.PropertyType;
// Collections need some customization when the collection attribute is present
if (typeof(ConfigurationElementCollection).IsAssignableFrom(propertyType)) {
ConfigurationCollectionAttribute attribCollection =
Attribute.GetCustomAttribute(propertyInformation,
typeof(ConfigurationCollectionAttribute)) as ConfigurationCollectionAttribute;
// If none on the property - see if there is an attribute on the collection type itself
if (attribCollection == null) {
attribCollection =
Attribute.GetCustomAttribute(propertyType,
typeof(ConfigurationCollectionAttribute)) as ConfigurationCollectionAttribute;
}
ConfigurationElementCollection coll = propertyInformation.GetValue(instance, null) as ConfigurationElementCollection;
if (coll == null) {
throw new ConfigurationErrorsException(SR.GetString(SR.Config_element_null_instance,
propertyInformation.Name, attribProperty.Name));
}
// If the attribute is found - get the collection instance and set the data from the attribute
if (attribCollection != null) {
if (attribCollection.AddItemName.IndexOf(',') == -1) {
coll.AddElementName = attribCollection.AddItemName;
}
coll.RemoveElementName = attribCollection.RemoveItemName;
coll.ClearElementName = attribCollection.ClearItemsName;
}
}
else if (typeof(ConfigurationElement).IsAssignableFrom(propertyType)) {
// Nested configuration element - handle recursively
object element = propertyInformation.GetValue(instance, null);
if (element == null) {
throw new ConfigurationErrorsException(SR.GetString(SR.Config_element_null_instance,
propertyInformation.Name,attribProperty.Name));
}
ApplyInstanceAttributes(element);
}
}
}
}
private static bool PropertiesFromType(Type type, out ConfigurationPropertyCollection result) {
ConfigurationPropertyCollection properties = (ConfigurationPropertyCollection)s_propertyBags[type];
result = null;
bool firstTimeInit = false;
if (properties == null) {
lock (s_propertyBags.SyncRoot) {
properties = (ConfigurationPropertyCollection)s_propertyBags[type];
if (properties == null) {
properties = CreatePropertyBagFromType(type);
s_propertyBags[type] = properties;
firstTimeInit = true;
}
}
}
result = properties;
return firstTimeInit;
}
private static ConfigurationPropertyCollection CreatePropertyBagFromType(Type type) {
Debug.Assert(type != null, "type != null");
// For ConfigurationElement derived classes - get the per-type validator
if (typeof(ConfigurationElement).IsAssignableFrom(type)) {
ConfigurationValidatorAttribute attribValidator = Attribute.GetCustomAttribute(type, typeof(ConfigurationValidatorAttribute)) as ConfigurationValidatorAttribute;
if (attribValidator != null) {
attribValidator.SetDeclaringType(type);
ConfigurationValidatorBase validator = attribValidator.ValidatorInstance;
if (validator != null) {
CachePerTypeValidator(type, validator);
}
}
}
ConfigurationPropertyCollection properties = new ConfigurationPropertyCollection();
foreach (PropertyInfo propertyInformation in type.GetProperties(BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.Instance)) {
ConfigurationProperty newProp = CreateConfigurationPropertyFromAttributes(propertyInformation);
if (newProp != null) {
properties.Add(newProp);
}
}
return properties;
}
private static ConfigurationProperty CreateConfigurationPropertyFromAttributes(PropertyInfo propertyInformation) {
Debug.Assert(propertyInformation != null, "propertyInformation != null");
ConfigurationProperty result = null;
ConfigurationPropertyAttribute attribProperty =
Attribute.GetCustomAttribute(propertyInformation,
typeof(ConfigurationPropertyAttribute)) as ConfigurationPropertyAttribute;
// If there is no ConfigurationProperty attrib - this is not considered a property
if (attribProperty != null) {
result = new ConfigurationProperty(propertyInformation);
}
// Handle some special cases of property types
if (result != null && typeof(ConfigurationElement).IsAssignableFrom(result.Type)) {
ConfigurationPropertyCollection unused = null;
PropertiesFromType(result.Type, out unused);
}
return result;
}
private static void CachePerTypeValidator( Type type, ConfigurationValidatorBase validator ) {
Debug.Assert((type != null) && ( validator != null));
Debug.Assert(typeof(ConfigurationElement).IsAssignableFrom(type));
// Use the same lock as the property bag lock since in the current implementation
// the only way to get to this method is through the code path that locks the property bag cache first ( see PropertiesFromType() )
// NOTE[ Thread Safety ]: Non-guarded access to static variable - since this code is called only from CreatePropertyBagFromType
// which in turn is done onle once per type and is guarded by the s_propertyBag.SyncRoot then this call is thread safe as well
if (s_perTypeValidators == null ) {
s_perTypeValidators = new Dictionary<Type,ConfigurationValidatorBase>();
}
// A type validator should be cached only once. If it isn't then attribute parsing is done more then once which should be avoided
Debug.Assert( !s_perTypeValidators.ContainsKey(type));
// Make sure the supplied validator supports validating this object
if (!validator.CanValidate(type)) {
throw new ConfigurationErrorsException(SR.GetString(SR.Validator_does_not_support_elem_type,
type.Name));
}
s_perTypeValidators.Add(type, validator);
}
private static void ApplyValidatorsRecursive(ConfigurationElement root) {
Debug.Assert(root != null);
// Apply the validator on 'root'
ApplyValidator(root);
// Apply validators on child elements ( note - we will do this only on already created child elements
// The non created ones will get their validators in the ctor
foreach (ConfigurationElement elem in root._values.ConfigurationElements) {
ApplyValidatorsRecursive(elem);
}
}
private static void ApplyValidator(ConfigurationElement elem) {
Debug.Assert(elem != null);
if ((s_perTypeValidators != null) && (s_perTypeValidators.ContainsKey(elem.GetType()))) {
elem._elementProperty = new ConfigurationElementProperty(s_perTypeValidators[ elem.GetType() ]);
}
}
protected void SetPropertyValue(ConfigurationProperty prop, object value, bool ignoreLocks) {
if (IsReadOnly()) {
throw new ConfigurationErrorsException(SR.GetString(SR.Config_base_read_only));
}
if ((ignoreLocks == false) &&
((_lockedAllExceptAttributesList != null && _lockedAllExceptAttributesList.HasParentElements && !_lockedAllExceptAttributesList.DefinedInParent(prop.Name)) ||
(_lockedAttributesList != null && (_lockedAttributesList.DefinedInParent(prop.Name) || _lockedAttributesList.DefinedInParent(LockAll))) ||
((_fItemLocked & ConfigurationValueFlags.Locked) != 0) &&
(_fItemLocked & ConfigurationValueFlags.Inherited) != 0)) {
throw new ConfigurationErrorsException(SR.GetString(SR.Config_base_attribute_locked, prop.Name));
}
_bModified = true;
// Run the new value through the validator to make sure its ok to store it
if (value != null) {
prop.Validate(value);
}
_values[prop.Name] = (value != null) ? value : s_nullPropertyValue;
}
protected internal virtual ConfigurationPropertyCollection Properties {
get {
ConfigurationPropertyCollection result = null;
if (PropertiesFromType(this.GetType(), out result)) {
ApplyInstanceAttributes(this); // Redundant but preserved to minimize code changes for Whidbey RTM
ApplyValidatorsRecursive(this);
}
return result;
}
}
internal ConfigurationValues Values {
get {
return _values;
}
}
internal PropertySourceInfo PropertyInfoInternal(string propertyName) {
return (PropertySourceInfo)_values.GetSourceInfo(propertyName);
}
internal string PropertyFileName(string propertyName) {
PropertySourceInfo p = (PropertySourceInfo)PropertyInfoInternal(propertyName);
if (p == null)
p = (PropertySourceInfo)PropertyInfoInternal(String.Empty); // Get the filename of the parent if prop is not there
if (p == null)
return String.Empty;
return p.FileName;
}
internal int PropertyLineNumber(string propertyName) {
PropertySourceInfo p = (PropertySourceInfo)PropertyInfoInternal(propertyName);
if (p == null)
p = (PropertySourceInfo)PropertyInfoInternal(String.Empty);
if (p == null)
return 0;
return p.LineNumber;
}
internal virtual void Dump(TextWriter tw) {
tw.WriteLine("Type: " + GetType().FullName);
foreach (PropertyInfo pi in GetType().GetProperties()) {
tw.WriteLine("{0}: {1}", pi.Name, pi.GetValue(this, null));
}
}
protected internal virtual void Unmerge(ConfigurationElement sourceElement,
ConfigurationElement parentElement,
ConfigurationSaveMode saveMode) {
if (sourceElement != null) {
bool hasAnyChildElements = false;
_lockedAllExceptAttributesList = sourceElement._lockedAllExceptAttributesList;
_lockedAllExceptElementsList = sourceElement._lockedAllExceptElementsList;
_fItemLocked = sourceElement._fItemLocked;
_lockedAttributesList = sourceElement._lockedAttributesList;
_lockedElementsList = sourceElement._lockedElementsList;
AssociateContext(sourceElement._configRecord);
if (parentElement != null) {
if (parentElement._lockedAttributesList != null)
_lockedAttributesList = UnMergeLockList(sourceElement._lockedAttributesList,
parentElement._lockedAttributesList, saveMode);
if (parentElement._lockedElementsList != null)
_lockedElementsList = UnMergeLockList(sourceElement._lockedElementsList,
parentElement._lockedElementsList, saveMode);
if (parentElement._lockedAllExceptAttributesList != null)
_lockedAllExceptAttributesList = UnMergeLockList(sourceElement._lockedAllExceptAttributesList,
parentElement._lockedAllExceptAttributesList, saveMode);
if (parentElement._lockedAllExceptElementsList != null)
_lockedAllExceptElementsList = UnMergeLockList(sourceElement._lockedAllExceptElementsList,
parentElement._lockedAllExceptElementsList, saveMode);
}
ConfigurationPropertyCollection props = Properties;
ConfigurationPropertyCollection collectionKeys = null;
// check for props not in bag from source
for (int index = 0; index < sourceElement.Values.Count; index++) {
string key = sourceElement.Values.GetKey(index);
object value = sourceElement.Values[index];
ConfigurationProperty prop = (ConfigurationProperty)sourceElement.Properties[key];
if (prop == null || (collectionKeys != null && !collectionKeys.Contains(prop.Name)))
continue;
if (prop.IsConfigurationElementType) {
hasAnyChildElements = true;
}
else {
if (value != s_nullPropertyValue) {
if (!props.Contains(key)) // this is for optional provider models keys
{
// _values[key] = value;
ConfigurationValueFlags valueFlags = sourceElement.Values.RetrieveFlags(key);
_values.SetValue(key, value, valueFlags, null);
props.Add(prop);
}
}
}
}
foreach (ConfigurationProperty prop in Properties) {
if (prop == null || (collectionKeys != null && !collectionKeys.Contains(prop.Name))) {
continue;
}
if (prop.IsConfigurationElementType) {
hasAnyChildElements = true;
}
else {
object value = sourceElement.Values[prop.Name];
// if the property is required or we are writing a full config make sure we have defaults
if ((prop.IsRequired == true || saveMode == ConfigurationSaveMode.Full) && (value == null || value == s_nullPropertyValue)) {
// If the default value is null, this means there wasnt a reasonable default for the value
// and there is nothing more we can do. Otherwise reset the value to the default
// Note: 'null' should be used as default for non-empty strings instead
// of the current practice to use String.Epmty
if (prop.DefaultValue != null) {
value = prop.DefaultValue; // need to make sure required properties are persisted
}
}
if (value != null && value != s_nullPropertyValue) {
object value2 = null;
if (parentElement != null) // Is there a parent
value2 = parentElement.Values[prop.Name]; // if so get it's value
if (value2 == null) // no parent use default
value2 = prop.DefaultValue;
// If changed and not same as parent write or required
switch (saveMode) {
case ConfigurationSaveMode.Minimal: {
if (!Object.Equals(value, value2) || prop.IsRequired == true)
_values[prop.Name] = value;
}
break;
// (value != null && value != s_nullPropertyValue) ||
case ConfigurationSaveMode.Modified: {
bool modified = sourceElement.Values.IsModified(prop.Name);
bool inherited = sourceElement.Values.IsInherited(prop.Name);
// update the value if the property is required, modified or it was not inherited
// Also update properties that ARE inherited when we are resetting the object
// as long as the property is not the same as the default value for the property
if ((prop.IsRequired || modified || !inherited) ||
(parentElement == null && inherited && !Object.Equals(value, value2))) {
_values[prop.Name] = value;
}
}
break;
case ConfigurationSaveMode.Full: {
if (value != null && value != s_nullPropertyValue)
_values[prop.Name] = value;
else
_values[prop.Name] = value2;
}
break;
}
}
}
}
if (hasAnyChildElements) {
foreach (ConfigurationProperty prop in Properties) {
if (prop.IsConfigurationElementType) {
ConfigurationElement pElem = (ConfigurationElement)((parentElement != null) ? parentElement[prop] : null);
ConfigurationElement childElement = (ConfigurationElement)this[prop];
if ((ConfigurationElement)sourceElement[prop] != null)
childElement.Unmerge((ConfigurationElement)sourceElement[prop],
pElem, saveMode);
}
}
}
}
}
protected internal virtual bool SerializeToXmlElement(XmlWriter writer, String elementName) {
if (_configRecord != null && _configRecord.TargetFramework != null) {
ConfigurationSection section = null;
if (_configRecord.SectionsStack.Count >0) {
section = _configRecord.SectionsStack.Peek() as ConfigurationSection;
}
if (section != null && !section.ShouldSerializeElementInTargetVersion(this, elementName, _configRecord.TargetFramework)) {
return false;
}
}
bool DataToWrite = _bDataToWrite;
// Don't write elements that are locked in the parent
if ((_lockedElementsList != null && _lockedElementsList.DefinedInParent(elementName)) ||
(_lockedAllExceptElementsList != null && _lockedAllExceptElementsList.HasParentElements && !_lockedAllExceptElementsList.DefinedInParent(elementName))) {
return DataToWrite;
}
if (SerializeElement(null, false) == true) // check if there is anything to write...
{
if (writer != null)
writer.WriteStartElement(elementName);
DataToWrite |= SerializeElement(writer, false);
if (writer != null)
writer.WriteEndElement();
}
return DataToWrite;
}
protected internal virtual bool SerializeElement(XmlWriter writer, bool serializeCollectionKey) {
PreSerialize(writer);
bool DataToWrite = _bDataToWrite;
bool hasAnyChildElements = false;
bool foundDefaultElement = false;
ConfigurationPropertyCollection props = Properties;
ConfigurationPropertyCollection collectionKeys = null;
for (int index = 0; index < _values.Count; index++) {
string key = _values.GetKey(index);
object value = _values[index];
ConfigurationProperty prop = (ConfigurationProperty)props[key];
if (prop == null || (collectionKeys != null && !collectionKeys.Contains(prop.Name))) {
continue;
}
if ( prop.IsVersionCheckRequired && _configRecord != null && _configRecord.TargetFramework != null) {
ConfigurationSection section = null;
if (_configRecord.SectionsStack.Count >0) {
section = _configRecord.SectionsStack.Peek() as ConfigurationSection;
}
if (section != null && !section.ShouldSerializePropertyInTargetVersion(prop, prop.Name, _configRecord.TargetFramework, this)) {
continue;
}
}
if (prop.IsConfigurationElementType) {
hasAnyChildElements = true;
}
else {
if ((_lockedAllExceptAttributesList != null && _lockedAllExceptAttributesList.HasParentElements && !_lockedAllExceptAttributesList.DefinedInParent(prop.Name)) ||
(_lockedAttributesList != null && _lockedAttributesList.DefinedInParent(prop.Name))) {
if (prop.IsRequired == true)
throw new ConfigurationErrorsException(SR.GetString(SR.Config_base_required_attribute_locked, prop.Name));
value = s_nullPropertyValue;
}
if (value != s_nullPropertyValue) {
if (serializeCollectionKey == false || prop.IsKey == true) {
string xmlValue = null;
// If this was an invalid string value and was cached - write it out as is
if (value is InvalidPropValue) {
xmlValue = ((InvalidPropValue)value).Value;
}
else {
prop.Validate(value);
xmlValue = prop.ConvertToString(value);
}
if ((xmlValue != null) && (writer != null)) {
if (prop.IsTypeStringTransformationRequired)
xmlValue = GetTransformedTypeString(xmlValue);
if (prop.IsAssemblyStringTransformationRequired)
xmlValue = GetTransformedAssemblyString(xmlValue);
writer.WriteAttributeString(prop.Name, xmlValue);
}
DataToWrite = DataToWrite || (xmlValue != null);
}
}
}
}
if (serializeCollectionKey == false) {
DataToWrite |= SerializeLockList(_lockedAttributesList, LockAttributesKey, writer);
DataToWrite |= SerializeLockList(_lockedAllExceptAttributesList, LockAllAttributesExceptKey, writer);
DataToWrite |= SerializeLockList(_lockedElementsList, LockElementsKey, writer);
DataToWrite |= SerializeLockList(_lockedAllExceptElementsList, LockAllElementsExceptKey, writer);
if ((_fItemLocked & ConfigurationValueFlags.Locked) != 0 &&
(_fItemLocked & ConfigurationValueFlags.Inherited) == 0 &&
(_fItemLocked & ConfigurationValueFlags.XMLParentInherited) == 0) {
DataToWrite = true;
if (writer != null)
writer.WriteAttributeString(LockItemKey, true.ToString().ToLower(CultureInfo.InvariantCulture));
}
}
if (hasAnyChildElements) {
for (int index = 0; index < _values.Count; index++) {
string key = _values.GetKey(index);
object value = _values[index];
ConfigurationProperty prop = (ConfigurationProperty)props[key];
// if we are writing a remove and the sub element is not part of the key don't write it.
if (serializeCollectionKey == false || prop.IsKey == true) {
if (value is ConfigurationElement) {
if (!((_lockedElementsList != null && _lockedElementsList.DefinedInParent(key)) ||
(_lockedAllExceptElementsList != null && _lockedAllExceptElementsList.HasParentElements && !_lockedAllExceptElementsList.DefinedInParent(key)))) {
ConfigurationElement elem = (ConfigurationElement)value;
if (prop.Name != ConfigurationProperty.DefaultCollectionPropertyName) {
DataToWrite |= elem.SerializeToXmlElement(writer, prop.Name);
}
else if (!foundDefaultElement) {
// Prevent the locks from serializing a second time since locks
// on a default collection serialize with their parent node
elem._lockedAttributesList = null;
elem._lockedAllExceptAttributesList = null;
elem._lockedElementsList = null;
elem._lockedAllExceptElementsList = null;
DataToWrite |= elem.SerializeElement(writer, false);
foundDefaultElement = true;
}
else {
throw new ConfigurationErrorsException(SR.GetString(SR.Config_base_element_cannot_have_multiple_child_elements, prop.Name));
}
}
}
}
}
}
return DataToWrite;
}
private bool SerializeLockList(ConfigurationLockCollection list, String elementKey, XmlWriter writer) {
StringBuilder sb;
sb = new StringBuilder();
if (list != null) {
foreach (string key in list) {
if (!list.DefinedInParent(key)) {
if (sb.Length != 0)
sb.Append(',');
sb.Append((string)key);
}
}
}
if (writer != null && sb.Length != 0)
writer.WriteAttributeString(elementKey, sb.ToString());
return (sb.Length != 0);
}
internal void ReportInvalidLock(string attribToLockTrim, ConfigurationLockCollectionType lockedType, ConfigurationValue value, String collectionProperties) {
StringBuilder sb;
sb = new StringBuilder();
// Add the collection properties when locking elements
if (!String.IsNullOrEmpty(collectionProperties) &&
((lockedType == ConfigurationLockCollectionType.LockedElements) || (lockedType == ConfigurationLockCollectionType.LockedElementsExceptionList))) {
if (sb.Length != 0)
sb.Append(',');
sb.Append(collectionProperties);
}
// construct a list of valid lockable properties
foreach (object _prop in Properties) {
ConfigurationProperty validProp = (ConfigurationProperty)_prop;
if (validProp.Name != LockAttributesKey &&
validProp.Name != LockAllAttributesExceptKey &&
validProp.Name != LockElementsKey &&
validProp.Name != LockAllElementsExceptKey
) {
if ((lockedType == ConfigurationLockCollectionType.LockedElements) ||
(lockedType == ConfigurationLockCollectionType.LockedElementsExceptionList)) {
if (typeof(ConfigurationElement).IsAssignableFrom(validProp.Type)) {
if (sb.Length != 0)
sb.Append(", ");
sb.Append("'");
sb.Append(validProp.Name);
sb.Append("'");
}
}
else {
if (!typeof(ConfigurationElement).IsAssignableFrom(validProp.Type)) {
if (sb.Length != 0)
sb.Append(", ");
sb.Append("'");
sb.Append(validProp.Name);
sb.Append("'");
}
}
}
}
string format = null;
if ((lockedType == ConfigurationLockCollectionType.LockedElements) ||
(lockedType == ConfigurationLockCollectionType.LockedElementsExceptionList)) {
if (value != null)
format = SR.GetString(SR.Config_base_invalid_element_to_lock);
else
format = SR.GetString(SR.Config_base_invalid_element_to_lock_by_add);
}
else {
if (value != null)
format = SR.GetString(SR.Config_base_invalid_attribute_to_lock);
else
format = SR.GetString(SR.Config_base_invalid_attribute_to_lock_by_add);
}
if (value != null)
throw new ConfigurationErrorsException(string.Format(CultureInfo.CurrentCulture, format, attribToLockTrim, sb.ToString()), value.SourceInfo.FileName, value.SourceInfo.LineNumber);
else
throw new ConfigurationErrorsException(string.Format(CultureInfo.CurrentCulture, format, attribToLockTrim, sb.ToString()));
}
private ConfigurationLockCollection ParseLockedAttributes(ConfigurationValue value, ConfigurationLockCollectionType lockType) {
// check that only actual properties are in the lock attribute
ConfigurationLockCollection localLockedAttributesList = new ConfigurationLockCollection(this, lockType);
string attributeList = (string)(value.Value);
if (string.IsNullOrEmpty(attributeList)) {
if (lockType == ConfigurationLockCollectionType.LockedAttributes)
throw new ConfigurationErrorsException(SR.GetString(SR.Empty_attribute, LockAttributesKey), value.SourceInfo.FileName, value.SourceInfo.LineNumber);
if (lockType == ConfigurationLockCollectionType.LockedElements)
throw new ConfigurationErrorsException(SR.GetString(SR.Empty_attribute, LockElementsKey), value.SourceInfo.FileName, value.SourceInfo.LineNumber);
if (lockType == ConfigurationLockCollectionType.LockedExceptionList)
throw new ConfigurationErrorsException(SR.GetString(SR.Config_empty_lock_attributes_except, LockAllAttributesExceptKey, LockAttributesKey), value.SourceInfo.FileName, value.SourceInfo.LineNumber);
if (lockType == ConfigurationLockCollectionType.LockedElementsExceptionList)
throw new ConfigurationErrorsException(SR.GetString(SR.Config_empty_lock_element_except, LockAllElementsExceptKey, LockElementsKey), value.SourceInfo.FileName, value.SourceInfo.LineNumber);
}
string[] attribsToLock = attributeList.Split(new char[] { ',', ':', ';' });
foreach (string attribToLock in attribsToLock) {
string attribToLockTrim = attribToLock.Trim();
if (!String.IsNullOrEmpty(attribToLockTrim)) {
// validate that the locks are good
if (!((lockType == ConfigurationLockCollectionType.LockedElements ||
lockType == ConfigurationLockCollectionType.LockedAttributes) &&
attribToLockTrim == LockAll)) {
ConfigurationProperty propToLock = Properties[attribToLockTrim];
if (propToLock == null || // if the prop does not exist
attribToLockTrim == LockAttributesKey || // or it is the lockattributes keyword
attribToLockTrim == LockAllAttributesExceptKey || // or it is the lockattributes keyword
attribToLockTrim == LockElementsKey || // or it is the lockelements keyword
(lockType != ConfigurationLockCollectionType.LockedElements && lockType != ConfigurationLockCollectionType.LockedElementsExceptionList &&
typeof(ConfigurationElement).IsAssignableFrom(propToLock.Type)) || // or if not locking elements but the property is a element
((lockType == ConfigurationLockCollectionType.LockedElements || lockType == ConfigurationLockCollectionType.LockedElementsExceptionList) &&
!typeof(ConfigurationElement).IsAssignableFrom(propToLock.Type)) // or if locking elements but the property is not an element
) {
// check to see if this is a collection and we are locking a collection element
ConfigurationElementCollection collection = this as ConfigurationElementCollection;
if (collection == null && Properties.DefaultCollectionProperty != null) // this is not a collection but it may contain a default collection
{
collection = this[Properties.DefaultCollectionProperty] as ConfigurationElementCollection;
}
if (collection == null ||
lockType == ConfigurationLockCollectionType.LockedAttributes || // If the collection type is not element then the lock is bogus
lockType == ConfigurationLockCollectionType.LockedExceptionList) {
ReportInvalidLock(attribToLockTrim, lockType, value, null);
}
else if (!collection.IsLockableElement(attribToLockTrim)) {
ReportInvalidLock(attribToLockTrim, lockType, value, collection.LockableElements);
}
}
if (propToLock != null && propToLock.IsRequired == true)
throw new ConfigurationErrorsException(SR.GetString(SR.Config_base_required_attribute_lock_attempt, propToLock.Name));
}
// concatenate the new attribute.
localLockedAttributesList.Add(attribToLockTrim, ConfigurationValueFlags.Default); // Mark as local
}
}
return localLockedAttributesList;
}
private StringCollection IntersectLockCollections(ConfigurationLockCollection Collection1, ConfigurationLockCollection Collection2) {
ConfigurationLockCollection smallCollection = Collection1.Count < Collection2.Count ? Collection1 : Collection2;
ConfigurationLockCollection largeCollection = Collection1.Count >= Collection2.Count ? Collection1 : Collection2;
StringCollection intersectionCollection = new StringCollection();
foreach (string key in smallCollection) {
if (largeCollection.Contains(key) || key == ElementTagName)
intersectionCollection.Add(key); // add the local copy
}
return intersectionCollection;
}
protected internal virtual void DeserializeElement(XmlReader reader, bool serializeCollectionKey) {
ConfigurationPropertyCollection props = Properties;
ConfigurationValue LockedAttributesList = null;
ConfigurationValue LockedAllExceptList = null;
ConfigurationValue LockedElementList = null;
ConfigurationValue LockedAllElementsExceptList = null;
bool ItemLockedLocally = false;
_bElementPresent = true;
ConfigurationElement defaultCollection = null;
ConfigurationProperty defaultCollectionProperty = props != null ? props.DefaultCollectionProperty : null;
if (defaultCollectionProperty != null) {
defaultCollection = (ConfigurationElement)this[defaultCollectionProperty];
}
// Process attributes
_elementTagName = reader.Name;
PropertySourceInfo rootInfo = new PropertySourceInfo(reader);
_values.SetValue(reader.Name, null, ConfigurationValueFlags.Modified, rootInfo);
_values.SetValue(DefaultCollectionPropertyName, defaultCollection, ConfigurationValueFlags.Modified, rootInfo);
if ((_lockedElementsList != null && (_lockedElementsList.Contains(reader.Name) ||
(_lockedElementsList.Contains(LockAll) && reader.Name != ElementTagName))) ||
(_lockedAllExceptElementsList != null && _lockedAllExceptElementsList.Count != 0 && !_lockedAllExceptElementsList.Contains(reader.Name)) ||
((_fItemLocked & ConfigurationValueFlags.Locked) != 0 && (_fItemLocked & ConfigurationValueFlags.Inherited) != 0)
) {
throw new ConfigurationErrorsException(SR.GetString(SR.Config_base_element_locked, reader.Name), reader);
}
if (reader.AttributeCount > 0) {
while (reader.MoveToNextAttribute()) {
String propertyName = reader.Name;
if ((_lockedAttributesList != null && (_lockedAttributesList.Contains(propertyName) || _lockedAttributesList.Contains(LockAll))) ||
(_lockedAllExceptAttributesList != null && !_lockedAllExceptAttributesList.Contains(propertyName))
) {
if (propertyName != LockAttributesKey && propertyName != LockAllAttributesExceptKey)
throw new ConfigurationErrorsException(SR.GetString(SR.Config_base_attribute_locked, propertyName), reader);
}
ConfigurationProperty prop = props != null ? props[propertyName] : null;
if (prop != null) {
if (serializeCollectionKey && !prop.IsKey) {
throw new ConfigurationErrorsException(SR.GetString(SR.Config_base_unrecognized_attribute, propertyName), reader);
}
_values.SetValue(propertyName,
DeserializePropertyValue(prop, reader),
ConfigurationValueFlags.Modified,
new PropertySourceInfo(reader));
} // if (deserializing a remove OR an add that does not handle optional attributes)
else if (propertyName == LockItemKey) {
try {
ItemLockedLocally = bool.Parse(reader.Value);
}
catch {
throw new ConfigurationErrorsException(SR.GetString(SR.Config_invalid_boolean_attribute, propertyName), reader);
}
}
else if (propertyName == LockAttributesKey) {
LockedAttributesList = new ConfigurationValue(reader.Value, ConfigurationValueFlags.Default, new PropertySourceInfo(reader));
}
else if (propertyName == LockAllAttributesExceptKey) {
LockedAllExceptList = new ConfigurationValue(reader.Value, ConfigurationValueFlags.Default, new PropertySourceInfo(reader));
}
else if (propertyName == LockElementsKey) {
LockedElementList = new ConfigurationValue(reader.Value, ConfigurationValueFlags.Default, new PropertySourceInfo(reader));
}
else if (propertyName == LockAllElementsExceptKey) {
LockedAllElementsExceptList = new ConfigurationValue(reader.Value, ConfigurationValueFlags.Default, new PropertySourceInfo(reader));
}
else if (serializeCollectionKey || !OnDeserializeUnrecognizedAttribute(propertyName, reader.Value)) {
throw new ConfigurationErrorsException(SR.GetString(SR.Config_base_unrecognized_attribute, propertyName), reader);
}
}
}
reader.MoveToElement();
// Check for nested elements.
try {
HybridDictionary nodeFound = new HybridDictionary();
if (!reader.IsEmptyElement) {
while (reader.Read()) {
if (reader.NodeType == XmlNodeType.Element) {
String propertyName = reader.Name;
CheckLockedElement(propertyName, null);
ConfigurationProperty prop = props != null ? props[propertyName] : null;
if (prop != null) {
if (prop.IsConfigurationElementType) {
//
if (nodeFound.Contains(propertyName))
throw new ConfigurationErrorsException(SR.GetString(SR.Config_base_element_cannot_have_multiple_child_elements, propertyName), reader);
nodeFound.Add(propertyName, propertyName);
ConfigurationElement childElement = (ConfigurationElement)this[prop];
childElement.DeserializeElement(reader, serializeCollectionKey);
// Validate the new element with the per-property Validator
// Note that the per-type validator for childElement has been already executed as part of Deserialize
ValidateElement(childElement, prop.Validator, false);
}
else {
throw new ConfigurationErrorsException(SR.GetString(SR.Config_base_property_is_not_a_configuration_element, propertyName), reader);
}
}
else if (!OnDeserializeUnrecognizedElement(propertyName, reader)) {
// Let the default collection, if there is one, handle this node.
if (defaultCollection == null ||
!defaultCollection.OnDeserializeUnrecognizedElement(propertyName, reader)) {
throw new ConfigurationErrorsException(SR.GetString(SR.Config_base_unrecognized_element_name, propertyName), reader);
}
}
}
else if (reader.NodeType == XmlNodeType.EndElement) {
break;
}
else if ((reader.NodeType == XmlNodeType.CDATA) || (reader.NodeType == XmlNodeType.Text)) {
throw new ConfigurationErrorsException(SR.GetString(SR.Config_base_section_invalid_content), reader);
}
}
}
EnsureRequiredProperties(serializeCollectionKey);
// Call the per-type validator for this object
ValidateElement(this, null, false);
}
catch (ConfigurationException e) {
// Catch the generic message from deserialization and include line info if necessary
if (e.Filename == null || e.Filename.Length == 0)
throw new ConfigurationErrorsException(e.Message, reader); // give it some info
else
throw e;
}
if (ItemLockedLocally) {
SetLocked();
_fItemLocked = ConfigurationValueFlags.Locked;
}
if (LockedAttributesList != null) {
if (_lockedAttributesList == null)
_lockedAttributesList = new ConfigurationLockCollection(this, ConfigurationLockCollectionType.LockedAttributes);
foreach (string key in ParseLockedAttributes(LockedAttributesList, ConfigurationLockCollectionType.LockedAttributes)) {
if (!_lockedAttributesList.Contains(key))
_lockedAttributesList.Add(key, ConfigurationValueFlags.Default); // add the local copy
else
_lockedAttributesList.Add(key, ConfigurationValueFlags.Modified | ConfigurationValueFlags.Inherited); // add the local copy
}
}
if (LockedAllExceptList != null) {
ConfigurationLockCollection newCollection = ParseLockedAttributes(LockedAllExceptList, ConfigurationLockCollectionType.LockedExceptionList);
if (_lockedAllExceptAttributesList == null) {
_lockedAllExceptAttributesList = new ConfigurationLockCollection(this, ConfigurationLockCollectionType.LockedExceptionList, String.Empty, newCollection);
_lockedAllExceptAttributesList.ClearSeedList(); // Prevent the list from thinking this was set by a parent.
}
StringCollection intersectionCollection = IntersectLockCollections(_lockedAllExceptAttributesList, newCollection);
/*
if (intersectionCollection.Count == 0) {
throw new ConfigurationErrorsException(SR.GetString(SR.Config_empty_lock_attributes_except_effective,
LockAllAttributesExceptKey,
LockedAllExceptList.Value,
LockAttributesKey),
LockedAllExceptList.SourceInfo.FileName,
LockedAllExceptList.SourceInfo.LineNumber);
}
*/
_lockedAllExceptAttributesList.ClearInternal(false);
foreach (string key in intersectionCollection) {
_lockedAllExceptAttributesList.Add(key, ConfigurationValueFlags.Default);
}
}
if (LockedElementList != null) {
if (_lockedElementsList == null)
_lockedElementsList = new ConfigurationLockCollection(this, ConfigurationLockCollectionType.LockedElements);
ConfigurationLockCollection localLockedElementList = ParseLockedAttributes(LockedElementList, ConfigurationLockCollectionType.LockedElements);
ConfigurationElementCollection collection = null;
if (props.DefaultCollectionProperty != null) // this is not a collection but it may contain a default collection
{
collection = this[props.DefaultCollectionProperty] as ConfigurationElementCollection;
if (collection != null && collection._lockedElementsList == null)
collection._lockedElementsList = _lockedElementsList;
}
foreach (string key in localLockedElementList) {
if (!_lockedElementsList.Contains(key)) {
_lockedElementsList.Add(key, ConfigurationValueFlags.Default); // add the local copy
ConfigurationProperty propToLock = Properties[key];
if (propToLock != null && typeof(ConfigurationElement).IsAssignableFrom(propToLock.Type)) {
((ConfigurationElement)this[key]).SetLocked();
}
if (key == LockAll) {
foreach (ConfigurationProperty prop in Properties) {
if (!string.IsNullOrEmpty(prop.Name) &&
prop.IsConfigurationElementType) {
((ConfigurationElement)this[prop]).SetLocked();
}
}
}
}
}
}
if (LockedAllElementsExceptList != null) {
ConfigurationLockCollection newCollection = ParseLockedAttributes(LockedAllElementsExceptList, ConfigurationLockCollectionType.LockedElementsExceptionList);
if (_lockedAllExceptElementsList == null) {
_lockedAllExceptElementsList = new ConfigurationLockCollection(this, ConfigurationLockCollectionType.LockedElementsExceptionList, _elementTagName, newCollection);
_lockedAllExceptElementsList.ClearSeedList();
}
StringCollection intersectionCollection = IntersectLockCollections(_lockedAllExceptElementsList, newCollection);
ConfigurationElementCollection collection = null;
if (props.DefaultCollectionProperty != null) // this is not a collection but it may contain a default collection
{
collection = this[props.DefaultCollectionProperty] as ConfigurationElementCollection;
if (collection != null && collection._lockedAllExceptElementsList == null)
collection._lockedAllExceptElementsList = _lockedAllExceptElementsList;
}
_lockedAllExceptElementsList.ClearInternal(false);
foreach (string key in intersectionCollection) {
if (!_lockedAllExceptElementsList.Contains(key) || key == ElementTagName)
_lockedAllExceptElementsList.Add(key, ConfigurationValueFlags.Default); // add the local copy
}
foreach (ConfigurationProperty prop in Properties) {
if (!(string.IsNullOrEmpty(prop.Name) || _lockedAllExceptElementsList.Contains(prop.Name)) &&
prop.IsConfigurationElementType) {
((ConfigurationElement)this[prop]).SetLocked();
}
}
}
// Make sure default collections use the same lock element lists
if (defaultCollectionProperty != null) {
defaultCollection = (ConfigurationElement)this[defaultCollectionProperty];
if (_lockedElementsList == null) {
_lockedElementsList = new ConfigurationLockCollection(this, ConfigurationLockCollectionType.LockedElements);
}
defaultCollection._lockedElementsList = _lockedElementsList;
if (_lockedAllExceptElementsList == null) {
_lockedAllExceptElementsList = new ConfigurationLockCollection(this, ConfigurationLockCollectionType.LockedElementsExceptionList, reader.Name);
_lockedAllExceptElementsList.ClearSeedList();
}
defaultCollection._lockedAllExceptElementsList = _lockedAllExceptElementsList;
}
// This has to be the last thing to execute
PostDeserialize();
}
private object DeserializePropertyValue(ConfigurationProperty prop, XmlReader reader) {
Debug.Assert(prop != null, "prop != null");
Debug.Assert(reader != null, "reader != null");
// By default we try to load (i.e. parse/validate ) all properties
// If a property value is invalid ( cannot be parsed or is not valid ) we will keep the value
// as string ( from the xml ) and will write it out unchanged if needed
// If the property value is needed by users the actuall exception will be thrown
string xmlValue = reader.Value;
object propertyValue = null;
try {
propertyValue = prop.ConvertFromString(xmlValue);
// Validate the loaded and converted value
prop.Validate(propertyValue);
}
catch (ConfigurationException ce) {
// If the error is incomplete - complete it :)
if (string.IsNullOrEmpty(ce.Filename)) {
ce = new ConfigurationErrorsException(ce.Message, reader);
}
// Cannot parse/validate the value. Keep it as string
propertyValue = new InvalidPropValue(xmlValue, ce);
}
catch {
// If this is an exception related to the parsing/validating the
// value ConfigurationErrorsException should be thrown instead.
// If not - the exception is ok to surface out of here
Debug.Fail("Unknown exception type thrown");
}
return propertyValue;
}
internal static void ValidateElement(ConfigurationElement elem, ConfigurationValidatorBase propValidator, bool recursive) {
// Validate a config element with the per-type validator when a per-property ( propValidator ) is not supplied
// or with the per-prop validator when the element ( elem ) is a child property of another configuration element
ConfigurationValidatorBase validator = propValidator;
if ((validator == null) && // Not a property - use the per-type validator
(elem.ElementProperty != null)) {
validator = elem.ElementProperty.Validator;
// Since ElementProperty can be overriden by derived classes we need to make sure that
// the validator supports the type of elem every time
if ((validator != null) && !validator.CanValidate(elem.GetType())) {
throw new ConfigurationErrorsException(SR.GetString(SR.Validator_does_not_support_elem_type, elem.GetType().Name));
}
}
try {
if (validator != null) {
validator.Validate(elem);
}
}
catch (ConfigurationException) {
// ConfigurationElement validators are allowed to throw ConfigurationErrorsException.
throw;
}
catch (Exception ex) {
throw new ConfigurationErrorsException(SR.GetString(SR.Validator_element_not_valid, elem._elementTagName, ex.Message));
}
if (recursive == true) {
// Validate collection items:
// Note: this is a bit of a hack - we will exploit the fact that this method is called with recursive == true only when serializing the top level section
// At deserializtion time the per-element validator for collection items will get executed as part of their deserialization logic
// However we dont perform validation in the serialization logic ( because at that time the object is unmerged and not all data is present )
// so we have to do that validation here.
if (elem is ConfigurationElementCollection) {
if (elem is ConfigurationElementCollection) {
IEnumerator it = ((ConfigurationElementCollection)elem).GetElementsEnumerator();
while( it.MoveNext() ) {
ValidateElement((ConfigurationElement)it.Current, null, true);
}
}
}
// Validate all child elements recursively
for (int index = 0; index < elem.Values.Count; index++) {
ConfigurationElement value = elem.Values[index] as ConfigurationElement;
if (value != null) {
// Run the per-type validator on the child element and proceed with validation in subelements
// Note we dont run the per-property validator here since we run those when the property value is set
ValidateElement(value, null, true); // per-type
}
}
}
}
private void EnsureRequiredProperties(bool ensureKeysOnly) {
ConfigurationPropertyCollection props = Properties;
// Make sure all required properties are here
if (props != null) {
foreach (ConfigurationProperty prop in props) {
// The property is required but no value was found
if (prop.IsRequired && !_values.Contains(prop.Name)) {
// Required properties can be ommited when we need only the keys to be there
if (!ensureKeysOnly || prop.IsKey) {
_values[prop.Name] = OnRequiredPropertyNotFound(prop.Name);
}
}
}
}
}
protected virtual object OnRequiredPropertyNotFound(string name) {
// Derivied classes can override this to return a value for a required property that is missing
// Here we treat this as an error though
throw new ConfigurationErrorsException(SR.GetString(SR.Config_base_required_attribute_missing, name),
PropertyFileName(name),
PropertyLineNumber(name));
}
protected virtual void PostDeserialize() {
// Please try to not add code in here
}
protected virtual void PreSerialize(XmlWriter writer) {
// Please try to not add code in here
}
protected virtual bool OnDeserializeUnrecognizedAttribute(String name, String value) {
return false;
}
protected virtual bool OnDeserializeUnrecognizedElement(String elementName, XmlReader reader) {
return false;
}
protected virtual string GetTransformedTypeString(string typeName)
{
if ( typeName == null || _configRecord == null || !_configRecord.TypeStringTransformerIsSet)
return typeName;
else
return _configRecord.TypeStringTransformer(typeName);
}
protected virtual string GetTransformedAssemblyString(string assemblyName)
{
if ( assemblyName == null || _configRecord == null || !_configRecord.AssemblyStringTransformerIsSet)
return assemblyName;
else
return _configRecord.AssemblyStringTransformer(assemblyName);
}
// Element
//
// Retrieve information specific to the element
//
public ElementInformation ElementInformation {
get {
if (_evaluationElement == null) {
_evaluationElement = new ElementInformation(this);
}
return _evaluationElement;
}
}
// EvaluationContext
//
// Retrieve information specific to the context of how we are
// being evaluated
//
protected ContextInformation EvaluationContext {
get {
if (_evalContext == null) {
if (_configRecord == null) {
// This is not associated with a context, so throw
// failure
throw new ConfigurationErrorsException(
SR.GetString(
SR.Config_element_no_context));
}
_evalContext = new ContextInformation(_configRecord);
}
return _evalContext;
}
}
internal protected virtual ConfigurationElementProperty ElementProperty {
get {
return _elementProperty;
}
}
internal ConfigurationLockCollection UnMergeLockList(
ConfigurationLockCollection sourceLockList,
ConfigurationLockCollection parentLockList,
ConfigurationSaveMode saveMode) {
if (sourceLockList.ExceptionList == false) {
switch (saveMode) {
case ConfigurationSaveMode.Modified: {
ConfigurationLockCollection tempLockList = new ConfigurationLockCollection(this, sourceLockList.LockType);
foreach (string lockedAttributeName in sourceLockList)
if (!parentLockList.Contains(lockedAttributeName) ||
sourceLockList.IsValueModified(lockedAttributeName)) {
tempLockList.Add(lockedAttributeName, ConfigurationValueFlags.Default);
}
return tempLockList;
}
case ConfigurationSaveMode.Minimal: {
ConfigurationLockCollection tempLockList = new ConfigurationLockCollection(this, sourceLockList.LockType);
foreach (string lockedAttributeName in sourceLockList)
if (!parentLockList.Contains(lockedAttributeName)) {
tempLockList.Add(lockedAttributeName, ConfigurationValueFlags.Default);
}
return tempLockList;
}
}
}
else {
// exception list write out the entire collection unless the entire collection
// came from the parent.
if (saveMode == ConfigurationSaveMode.Modified || saveMode == ConfigurationSaveMode.Minimal) {
bool sameAsParent = false;
if (sourceLockList.Count == parentLockList.Count) {
sameAsParent = true;
foreach (string lockedAttributeName in sourceLockList) {
if (!parentLockList.Contains(lockedAttributeName) ||
(sourceLockList.IsValueModified(lockedAttributeName) &&
saveMode == ConfigurationSaveMode.Modified)) {
sameAsParent = false;
}
}
}
if (sameAsParent == true) {
return null;
}
}
}
return sourceLockList;
}
//
// Return true if an attribute is one of our reserved locking attributes,
// false otherwise.
//
internal static bool IsLockAttributeName(string name) {
// optimize for common case that attribute name does not start with "lock"
if (!StringUtil.StartsWith(name, "lock")) {
return false;
}
foreach (string lockAttributeName in s_lockAttributeNames) {
if (name == lockAttributeName) {
return true;
}
}
return false;
}
protected bool HasContext {
get {
return _configRecord != null;
}
}
public Configuration CurrentConfiguration {
get {
return (_configRecord==null) ? null : _configRecord.CurrentConfiguration;
}
}
}
}