//------------------------------------------------------------------------------
//
// Copyright (c) Microsoft Corporation. All rights reserved.
//
//------------------------------------------------------------------------------
namespace System.Web.Configuration {
using System;
using System.Xml;
using System.Configuration;
using System.Collections.Specialized;
using System.Collections;
using System.Globalization;
using System.IO;
using System.Text;
using System.ComponentModel;
using System.Web.Hosting;
using System.Web.Util;
using System.Web.Configuration;
using System.Web.Management;
using System.Web.Compilation;
using System.Security.Permissions;
internal class HealthMonitoringSectionHelper {
static HealthMonitoringSectionHelper s_helper;
static RuleInfoComparer s_ruleInfoComparer = new RuleInfoComparer();
HealthMonitoringSection _section;
internal ProviderInstances _providerInstances;
internal Hashtable _customEvaluatorInstances;
internal ArrayList _ruleInfos;
bool _enabled;
// Cached matched rules for system events. For details, see comments for s_eventArrayDimensionSizes
// in WebEventCodes.cs
static ArrayList[,] _cachedMatchedRules;
#if DBG
// Because we assume no two different event type will use the same event code, in debug mode
// we use this hashtable to make sure our assumption is true.
Hashtable _cachedTypeOfMatchedRulesSystem = new Hashtable();
#endif
// Cached matched rules based on WebBaseEvent Hashcode, and is for non-system events.
Hashtable _cachedMatchedRulesForCustomEvents;
static internal HealthMonitoringSectionHelper GetHelper() {
if (s_helper == null) {
s_helper = new HealthMonitoringSectionHelper();
}
return s_helper;
}
HealthMonitoringSectionHelper() {
// Handle config exceptions so we can still log messages to the event log.
try {
_section = RuntimeConfig.GetAppConfig().HealthMonitoring;
}
catch(Exception e) {
// If InitializationException has not already been set, then this exception
// is happening because the section has an error.
// By setting InitializationException, we allow the exception to be displayed in the response.
// If InitializationException is already set, ignore this exception so we can
// display the original in the response.
if (HttpRuntime.InitializationException == null) {
HttpRuntime.InitializationException = e;
}
_section = RuntimeConfig.GetAppLKGConfig().HealthMonitoring;
// WOS 1965670: if we fail to get the section throw the previous error
if (_section == null) {
throw;
}
}
_enabled = _section.Enabled;
if (!_enabled) {
return;
}
// First run some basic sanity check
BasicSanityCheck();
// Init some class members
_ruleInfos = new ArrayList();
_customEvaluatorInstances = new Hashtable();
_providerInstances = new ProviderInstances(_section);
_cachedMatchedRulesForCustomEvents = new Hashtable(new WebBaseEventKeyComparer());
_cachedMatchedRules = new ArrayList[WebEventCodes.GetEventArrayDimensionSize(0),
WebEventCodes.GetEventArrayDimensionSize(1)];
BuildRuleInfos();
_providerInstances.CleanupUninitProviders();
}
internal bool Enabled {
get { return _enabled; }
}
internal HealthMonitoringSection HealthMonitoringSection {
get { return _section; }
}
[FileIOPermission(SecurityAction.Assert, Unrestricted = true)]
void BasicSanityCheck() {
Type type;
foreach (ProviderSettings providerSettings in _section.Providers) {
// Make sure the type is valid.
type = ConfigUtil.GetType(providerSettings.Type, "type", providerSettings);
// Make sure the type support WebEventProvider
HandlerBase.CheckAssignableType(providerSettings.ElementInformation.Properties["type"].Source,
providerSettings.ElementInformation.Properties["type"].LineNumber,
typeof(WebEventProvider), type);
}
foreach (EventMappingSettings eventMappingSettings in _section.EventMappings) {
// Make sure the type is valid.
type = ConfigUtil.GetType(eventMappingSettings.Type, "type", eventMappingSettings);
// Make sure startEventCode <= endEventCode
if (!(eventMappingSettings.StartEventCode <= eventMappingSettings.EndEventCode)) {
string attribute;
// We don't know which one was specified unless we test it
attribute = "startEventCode";
if (eventMappingSettings.ElementInformation.Properties[attribute].LineNumber == 0) {
attribute = "endEventCode";
Debug.Assert(eventMappingSettings.ElementInformation.Properties[attribute].LineNumber != 0,
"eventMappingSettings.ElementInformation.Properties[attribute].LineNumber != 0");
}
throw new ConfigurationErrorsException(
SR.GetString(SR.Event_name_invalid_code_range),
eventMappingSettings.ElementInformation.Properties[attribute].Source, eventMappingSettings.ElementInformation.Properties[attribute].LineNumber);
}
// Make sure the type support WebBaseEvent
HandlerBase.CheckAssignableType(eventMappingSettings.ElementInformation.Properties["type"].Source,
eventMappingSettings.ElementInformation.Properties["type"].LineNumber,
typeof(System.Web.Management.WebBaseEvent), type);
// It's a valid type. Might as well save it.
eventMappingSettings.RealType = type;
}
foreach (RuleSettings rule in _section.Rules) {
// Go thru all the Rules, and make sure all referenced provider, eventName
// and profile exist.
string provider = rule.Provider;
if (!String.IsNullOrEmpty(provider)) {
ProviderSettings providerSettings = _section.Providers[provider];
if (providerSettings == null) {
throw new ConfigurationErrorsException(
SR.GetString(SR.Health_mon_provider_not_found, provider),
rule.ElementInformation.Properties["provider"].Source,
rule.ElementInformation.Properties["provider"].LineNumber);
}
}
string profile = rule.Profile;
if (!String.IsNullOrEmpty(profile)) {
if (_section.Profiles[profile] == null) {
throw new ConfigurationErrorsException(
SR.GetString(SR.Health_mon_profile_not_found, profile),
rule.ElementInformation.Properties["profile"].Source,
rule.ElementInformation.Properties["profile"].LineNumber);
}
}
if (_section.EventMappings[rule.EventName] == null) {
throw new ConfigurationErrorsException(
SR.GetString(SR.Event_name_not_found, rule.EventName),
rule.ElementInformation.Properties["eventName"].Source, rule.ElementInformation.Properties["eventName"].LineNumber);
}
}
}
void DisplayRuleInfo(RuleInfo ruleInfo) {
#if DEBUG
Debug.Trace("BuildRuleInfos", "====================== Rule Info =======================");
Debug.Trace("BuildRuleInfos", "name:" + ruleInfo._ruleSettings.Name);
Debug.Trace("BuildRuleInfos", "type:" + ruleInfo._eventMappingSettings.RealType.Name);
Debug.Trace("BuildRuleInfos", "minInstances:" + ruleInfo._minInstances);
Debug.Trace("BuildRuleInfos", "maxLimit:" + ruleInfo._maxLimit);
Debug.Trace("BuildRuleInfos", "minInterval:" + ruleInfo._minInterval);
Debug.Trace("BuildRuleInfos", "provider:" + ruleInfo._ruleSettings.Provider);
Debug.Trace("BuildRuleInfos", "referenced provider:" + (ruleInfo._referencedProvider == null ? String.Empty : ruleInfo._referencedProvider.GetType().Name));
Debug.Trace("BuildRuleInfos", "=========================================================");
#endif
}
void BuildRuleInfos() {
Debug.Trace("BuildRuleInfos", "BuildRuleInfos called");
// Each ruleInfo is an object that takes the information
// stored in a ruleSettings and merge it with values from profileSettings.
// At the end, we'll sort the rules based on type (most specific type last)
foreach (RuleSettings ruleSettings in _section.Rules) {
RuleInfo ruleInfo = CreateRuleInfo(ruleSettings);
DisplayRuleInfo(ruleInfo);
_ruleInfos.Add(ruleInfo);
}
_ruleInfos.Sort(s_ruleInfoComparer);
}
RuleInfo CreateRuleInfo(RuleSettings ruleSettings) {
RuleInfo ruleInfo = new RuleInfo(ruleSettings, _section);
// Inherit values from profile
MergeValuesWithProfile(ruleInfo);
// Find out which provider it's referencing
InitReferencedProvider(ruleInfo);
// Initialize the cutom evaluator type
InitCustomEvaluator(ruleInfo);
return ruleInfo;
}
void InitReferencedProvider(RuleInfo ruleInfo) {
String providerName;
WebEventProvider provider;
Debug.Assert(ruleInfo._referencedProvider == null, "ruleInfo._referencedProvider == null");
providerName = ruleInfo._ruleSettings.Provider;
if (String.IsNullOrEmpty(providerName)) {
return;
}
provider = _providerInstances[providerName];
Debug.Assert(provider != null, "provider != null");
ruleInfo._referencedProvider = provider;
}
void MergeValuesWithProfile(RuleInfo ruleInfo) {
ProfileSettings profileSettings = null;
if (ruleInfo._ruleSettings.ElementInformation.Properties["profile"].ValueOrigin != PropertyValueOrigin.Default) {
profileSettings = _section.Profiles[ruleInfo._ruleSettings.Profile];
Debug.Assert(profileSettings != null, "profileSettings != null");
}
if (profileSettings != null && ruleInfo._ruleSettings.ElementInformation.Properties["minInstances"].ValueOrigin == PropertyValueOrigin.Default) {
ruleInfo._minInstances = profileSettings.MinInstances;
}
else {
ruleInfo._minInstances = ruleInfo._ruleSettings.MinInstances;
}
if (profileSettings != null && ruleInfo._ruleSettings.ElementInformation.Properties["maxLimit"].ValueOrigin == PropertyValueOrigin.Default) {
ruleInfo._maxLimit = profileSettings.MaxLimit;
}
else {
ruleInfo._maxLimit = ruleInfo._ruleSettings.MaxLimit;
}
if (profileSettings != null && ruleInfo._ruleSettings.ElementInformation.Properties["minInterval"].ValueOrigin == PropertyValueOrigin.Default) {
ruleInfo._minInterval = profileSettings.MinInterval;
}
else {
ruleInfo._minInterval = ruleInfo._ruleSettings.MinInterval;
}
if (profileSettings != null && ruleInfo._ruleSettings.ElementInformation.Properties["custom"].ValueOrigin == PropertyValueOrigin.Default) {
ruleInfo._customEvaluator = profileSettings.Custom;
ruleInfo._customEvaluatorConfig = profileSettings;
}
else {
ruleInfo._customEvaluator = ruleInfo._ruleSettings.Custom;
ruleInfo._customEvaluatorConfig = ruleInfo._ruleSettings;
}
}
void InitCustomEvaluator(RuleInfo ruleInfo) {
string customEvaluator = ruleInfo._customEvaluator;
if (customEvaluator == null ||
customEvaluator.Trim().Length == 0) {
ruleInfo._customEvaluatorType = null;
return;
}
ruleInfo._customEvaluatorType = ConfigUtil.GetType(ruleInfo._customEvaluator,
"custom", ruleInfo._customEvaluatorConfig);
// Make sure the type support WebBaseEvent
HandlerBase.CheckAssignableType(ruleInfo._customEvaluatorConfig.ElementInformation.Properties["custom"].Source,
ruleInfo._customEvaluatorConfig.ElementInformation.Properties["custom"].LineNumber,
typeof(System.Web.Management.IWebEventCustomEvaluator), ruleInfo._customEvaluatorType);
// Create a public instance of the custom evaluator
if (_customEvaluatorInstances[ruleInfo._customEvaluatorType] == null) {
_customEvaluatorInstances[ruleInfo._customEvaluatorType] = HttpRuntime.CreatePublicInstance(ruleInfo._customEvaluatorType);
}
}
// Find the corresponding array of RuleInfo based on the fired event
internal ArrayList FindFiringRuleInfos(Type eventType, int eventCode) {
ArrayList foundFiringRuleInfos;
bool systemEvent = eventCode < WebEventCodes.WebExtendedBase;
CustomWebEventKey customWebEventKey = null;
object lockObject;
int index0 = 0, index1 = 0;
#if DBG
if (systemEvent) {
Type type;
type = (Type)_cachedTypeOfMatchedRulesSystem[eventCode];
if (type == null) {
lock(_cachedTypeOfMatchedRulesSystem) {
type = (Type)_cachedTypeOfMatchedRulesSystem[eventCode];
if (type == null) {
_cachedTypeOfMatchedRulesSystem[eventCode] = eventType;
}
}
}
if (type != null) {
Debug.Assert(type == eventType,
"For system events, we assume each event code will map only to one event type. " +
"Eventcode= " + eventCode + "; stored type= " + type.ToString() +
"; raised event type= " + eventType);
}
}
#endif
// First, we look at the cache to see if we find the array.
if (systemEvent) {
WebEventCodes.GetEventArrayIndexsFromEventCode(eventCode, out index0, out index1);
foundFiringRuleInfos = _cachedMatchedRules[index0, index1];
}
else {
customWebEventKey = new CustomWebEventKey(eventType, eventCode);
foundFiringRuleInfos = (ArrayList)_cachedMatchedRulesForCustomEvents[customWebEventKey];
}
if (foundFiringRuleInfos != null) {
return foundFiringRuleInfos;
}
if (systemEvent) {
lockObject = _cachedMatchedRules;
}
else {
lockObject = _cachedMatchedRulesForCustomEvents;
}
lock (lockObject) {
if (systemEvent) {
foundFiringRuleInfos = _cachedMatchedRules[index0, index1];
}
else {
Debug.Assert(customWebEventKey != null);
foundFiringRuleInfos = (ArrayList)_cachedMatchedRulesForCustomEvents[customWebEventKey];
}
if (foundFiringRuleInfos != null) {
return foundFiringRuleInfos;
}
// Not found in cache.
ArrayList matchedRules = new ArrayList();
// Go thru the sorted ruleInfo array and look for matching ruleInfo,
// starting from the most specific type.
for (int i = _ruleInfos.Count - 1; i >= 0; i--) {
RuleInfo curRule = (RuleInfo)_ruleInfos[i];
// Now see if the current rule matches the raised event
if (curRule.Match(eventType, eventCode)) {
matchedRules.Add(new FiringRuleInfo(curRule));
}
}
// Then for each matched rule, we need to figure out if the provider it
// uses is also used by other rules. We need this info because if multiple rules are
// using the same provider, we fire the event to the provider only once.
int count = matchedRules.Count;
for (int i = 0; i < count; i++) {
FiringRuleInfo info1 = (FiringRuleInfo)matchedRules[i];
if (info1._ruleInfo._referencedProvider != null) {
for (int j = i + 1; j < count; j++) {
FiringRuleInfo info2 = (FiringRuleInfo)matchedRules[j];
if (info2._ruleInfo._referencedProvider != null && // ignore null-provider
info2._indexOfFirstRuleInfoWithSameProvider == -1 && // ignore rules that were marked already
info1._ruleInfo._referencedProvider == info2._ruleInfo._referencedProvider) { // they are pointing to the same provider
// We'll remember the index of the first rule info that share the same
// provider. For details on how this index is used, please see
// WebBaseEvent.RaiseInternal.
if (info1._indexOfFirstRuleInfoWithSameProvider == -1) {
info1._indexOfFirstRuleInfoWithSameProvider = i;
}
info2._indexOfFirstRuleInfoWithSameProvider = i;
}
}
}
}
#if DBG
Debug.Trace("FindRuleInfos", "------------------------------------------------");
Debug.Trace("FindRuleInfos", "Find ruleInfos for event with type=" + eventType.ToString() +
", EventCode=" + eventCode);
foreach(FiringRuleInfo info in matchedRules) {
Debug.Trace("FindRuleInfos", "Provider=" + info._ruleInfo._ruleSettings.Provider +
"; eventNameType=" + info._ruleInfo._eventMappingSettings.RealType.ToString() +
"; _indexOfFirstRuleInfoWithSameProvider=" + info._indexOfFirstRuleInfoWithSameProvider);
}
Debug.Trace("FindRuleInfos", "------------------------------------------------");
#endif
// save matchedRules in the cache
if (systemEvent) {
_cachedMatchedRules[index0, index1] = matchedRules;
}
else {
Debug.Assert(customWebEventKey != null);
_cachedMatchedRulesForCustomEvents[customWebEventKey] = matchedRules;
}
return matchedRules;
}
}
internal class RuleInfo {
internal string _customEvaluator;
internal ConfigurationElement _customEvaluatorConfig;
// The following properties are cached here for performance reason
internal int _minInstances;
internal int _maxLimit;
internal TimeSpan _minInterval; // in seconds
internal RuleSettings _ruleSettings;
internal WebEventProvider _referencedProvider;
internal Type _customEvaluatorType;
internal EventMappingSettings _eventMappingSettings;
internal RuleFiringRecord _ruleFiringRecord;
internal RuleInfo(RuleSettings ruleSettings, HealthMonitoringSection section) {
_eventMappingSettings = section.EventMappings[ruleSettings.EventName];
_ruleSettings = ruleSettings;
_ruleFiringRecord = new RuleFiringRecord(this);
}
internal bool Match(Type eventType, int eventCode) {
// Fail if the type doesn't match.
if (!(eventType.Equals(_eventMappingSettings.RealType) ||
eventType.IsSubclassOf(_eventMappingSettings.RealType))) {
return false;
}
// Fail if the event code doesn't match
if (!(_eventMappingSettings.StartEventCode <= eventCode &&
eventCode <= _eventMappingSettings.EndEventCode)) {
return false;
}
return true;
}
}
internal class FiringRuleInfo {
internal RuleInfo _ruleInfo;
internal int _indexOfFirstRuleInfoWithSameProvider;
internal FiringRuleInfo(RuleInfo ruleInfo) {
_ruleInfo = ruleInfo;
_indexOfFirstRuleInfoWithSameProvider = -1;
}
}
internal class ProviderInstances {
internal Hashtable _instances; // case-insensitive because the providers collection is too.
[PermissionSet(SecurityAction.Assert, Unrestricted = true)]
internal ProviderInstances(HealthMonitoringSection section) {
// Build the array of providers
// Don't create an instance yet, but only store the providerInfo in the HashTable.
_instances = CollectionsUtil.CreateCaseInsensitiveHashtable(section.Providers.Count);
foreach (object obj in section.Providers) {
ProviderSettings settings = (ProviderSettings)obj;
// Please note we are storing the ProviderSettings in the hashtable.
// But if we create an instance of that provider, we will replace
// that string with a provider object.
_instances.Add(settings.Name, settings);
}
}
WebEventProvider GetProviderInstance(string providerName) {
WebEventProvider provider;
object o;
o = _instances[providerName];
if (o == null) {
return null;
}
ProviderSettings providerSettings = o as ProviderSettings;
if (providerSettings != null) {
// If what we got is still a ProviderSettings, it means we haven't created an instance
// of it yet.
Type type;
string typeName = providerSettings.Type;
type = BuildManager.GetType(typeName, false);
Debug.Assert(type != null, "type != null");
if (typeof(IInternalWebEventProvider).IsAssignableFrom(type)) {
provider = (WebEventProvider)HttpRuntime.CreateNonPublicInstance(type);
}
else {
provider = (WebEventProvider)HttpRuntime.CreatePublicInstance(type);
}
using (new ProcessImpersonationContext()) {
try {
provider.Initialize(providerSettings.Name, providerSettings.Parameters);
}
catch (ConfigurationErrorsException) {
throw;
}
catch (ConfigurationException e) {
throw new ConfigurationErrorsException(e.Message, providerSettings.ElementInformation.Properties["type"].Source,
providerSettings.ElementInformation.Properties["type"].LineNumber);
}
catch {
throw;
}
}
Debug.Trace("ProviderInstances", "Create a provider instance: " +
"name=" + providerSettings.Name + ";type=" + typeName);
_instances[providerName] = provider;
}
else {
provider = o as WebEventProvider;
Debug.Assert(provider != null, "provider != null");
}
return provider;
}
internal WebEventProvider this[String name] {
get {
return GetProviderInstance(name);
}
}
// Cleanup each provider for which we have NOT created an instance.
internal void CleanupUninitProviders() {
ArrayList list = new ArrayList();
foreach (DictionaryEntry de in _instances) {
if (de.Value is ProviderSettings) {
list.Add(de.Key);
}
}
foreach (object o in list) {
Debug.Trace("ProviderInstances", "Remove " + (string)o + " from providers");
_instances.Remove(o);
}
}
internal bool ContainsKey(string name) {
return _instances.ContainsKey(name);
}
public IDictionaryEnumerator GetEnumerator() {
return _instances.GetEnumerator();
}
}
}
}