using System.Collections;
using System.Collections.Specialized;
using System.Reflection;
+using System.Web.Util;
using System.Xml;
using System.Configuration;
using System.Configuration.Internal;
{
#if !TARGET_J2EE
static IInternalConfigConfigurationFactory configFactory;
- static Hashtable configurations = new Hashtable ();
+ static Hashtable configurations = Hashtable.Synchronized (new Hashtable ());
+ static Hashtable sectionCache = new Hashtable (StringComparer.OrdinalIgnoreCase);
#else
+ const string AppSettingsKey = "WebConfigurationManager.AppSettings";
static internal IInternalConfigConfigurationFactory configFactory
{
get{
lock (AppDomain.CurrentDomain){
object initialized = AppDomain.CurrentDomain.GetData("WebConfigurationManager.configurations.initialized");
if (initialized == null){
- table = new Hashtable();
+ table = Hashtable.Synchronized (new Hashtable (StringComparer.OrdinalIgnoreCase));
configurations = table;
}
}
AppDomain.CurrentDomain.SetData("WebConfigurationManager.configurations.initialized", true);
}
}
+
+ static Hashtable sectionCache
+ {
+ get
+ {
+ Hashtable sectionCache = (Hashtable) AppDomain.CurrentDomain.GetData ("sectionCache");
+ if (sectionCache == null) {
+ sectionCache = new Hashtable (StringComparer.OrdinalIgnoreCase);
+ AppDomain.CurrentDomain.SetData ("sectionCache", sectionCache);
+ }
+ return sectionCache;
+ }
+ set
+ {
+ AppDomain.CurrentDomain.SetData ("sectionCache", value);
+ }
+ }
#endif
- static internal ArrayList extra_assemblies = null;
+ static ArrayList extra_assemblies = null;
static internal ArrayList ExtraAssemblies {
get {
if (extra_assemblies == null)
return extra_assemblies;
}
}
+
+ static bool hasConfigErrors = false;
+ static object hasConfigErrorsLock = new object ();
+ static internal bool HasConfigErrors {
+ get {
+ lock (hasConfigErrorsLock) {
+ return hasConfigErrors;
+ }
+ }
+ }
static WebConfigurationManager ()
{
return ConfigurationManager.OpenMachineConfiguration ();
}
- [MonoTODO ("need to handle locationSubPath")]
+ [MonoLimitation ("locationSubPath is not handled")]
public static _Configuration OpenMachineConfiguration (string locationSubPath)
{
return OpenMachineConfiguration ();
}
- [MonoTODO("Mono does not support remote configuration")]
+ [MonoLimitation("Mono does not support remote configuration")]
public static _Configuration OpenMachineConfiguration (string locationSubPath,
string server)
{
throw new NotSupportedException ("Mono doesn't support remote configuration");
}
- [MonoTODO("Mono does not support remote configuration")]
+ [MonoLimitation("Mono does not support remote configuration")]
public static _Configuration OpenMachineConfiguration (string locationSubPath,
string server,
IntPtr userToken)
throw new NotSupportedException ("Mono doesn't support remote configuration");
}
- [MonoTODO("Mono does not support remote configuration")]
+ [MonoLimitation("Mono does not support remote configuration")]
public static _Configuration OpenMachineConfiguration (string locationSubPath,
string server,
string userName,
public static _Configuration OpenWebConfiguration (string path, string site, string locationSubPath, string server, string userName, string password)
{
- if (path == null)
- path = "";
+ if (path == null || path.Length == 0)
+ path = "/";
- string basePath = GetBasePath (path);
_Configuration conf;
-
- lock (configurations) {
- conf = (_Configuration) configurations [basePath];
- if (conf == null) {
- conf = ConfigurationFactory.Create (typeof(WebConfigurationHost), null, basePath, site, locationSubPath, server, userName, password);
- configurations [basePath] = conf;
- }
- }
- if (basePath.Length < path.Length) {
-
- // If the path has a file name, look for a location specific configuration
-
- int dif = path.Length - basePath.Length;
- string file = path.Substring (path.Length - dif);
- int i=0;
- while (i < file.Length && file [i] == '/')
- i++;
- if (i != 0)
- file = file.Substring (i);
-
- if (file.Length != 0) {
- foreach (ConfigurationLocation loc in conf.Locations) {
- if (loc.Path == file)
- return loc.OpenConfiguration ();
+ conf = (_Configuration) configurations [path];
+ if (conf == null) {
+ try {
+ conf = ConfigurationFactory.Create (typeof (WebConfigurationHost), null, path, site, locationSubPath, server, userName, password);
+ configurations [path] = conf;
+ } catch (Exception ex) {
+ lock (hasConfigErrorsLock) {
+ hasConfigErrors = true;
}
+ throw ex;
}
}
return conf;
return OpenMappedMachineConfiguration (fileMap);
}
- public static object GetSection (string sectionName)
+ internal static object SafeGetSection (string sectionName, Type configSectionType)
{
- _Configuration c;
- if (HttpContext.Current != null
- && HttpContext.Current.Request != null)
- c = OpenWebConfiguration (HttpContext.Current.Request.Path);
- else
- c = OpenWebConfiguration (HttpRuntime.AppDomainAppVirtualPath);
-
- if (c == null)
+ try {
+ return GetSection (sectionName);
+ } catch (Exception) {
+ if (configSectionType != null)
+ return Activator.CreateInstance (configSectionType);
return null;
- else
- return c.GetSection (sectionName);
+ }
}
-
- public static object GetSection (string sectionName, string path)
+
+ internal static object SafeGetSection (string sectionName, string path, Type configSectionType)
{
try {
- _Configuration c = OpenWebConfiguration (path);
- return c.GetSection (sectionName);
- }
- catch {
+ return GetSection (sectionName, path);
+ } catch (Exception) {
+ if (configSectionType != null)
+ return Activator.CreateInstance (configSectionType);
return null;
}
}
- static _Configuration GetWebApplicationConfiguration ()
+ public static object GetSection (string sectionName)
{
- _Configuration config;
+ return GetSection (sectionName, GetCurrentPath (HttpContext.Current));
+ }
- if (HttpContext.Current == null
- || HttpContext.Current.Request == null
- || HttpContext.Current.Request.ApplicationPath == null
- || HttpContext.Current.Request.ApplicationPath == "") {
- config = OpenWebConfiguration ("");
- }
- else {
- config = OpenWebConfiguration (HttpContext.Current.Request.ApplicationPath);
+ public static object GetSection (string sectionName, string path)
+ {
+ object cachedSection = sectionCache [GetSectionCacheKey (sectionName, path)];
+ if (cachedSection != null)
+ return cachedSection;
+
+ _Configuration c = OpenWebConfiguration (path);
+ ConfigurationSection section = c.GetSection (sectionName);
+
+ if (section == null)
+ return null;
+
+#if TARGET_J2EE
+ object value = get_runtime_object.Invoke (section, new object [0]);
+ if (String.CompareOrdinal ("appSettings", sectionName) == 0) {
+ NameValueCollection collection;
+ collection = new KeyValueMergedCollection (HttpContext.Current, (NameValueCollection) value);
+ value = collection;
}
- return config;
+ AddSectionToCache (GetSectionCacheKey (sectionName, path), value);
+ return value;
+#else
+ object value = SettingsMappingManager.MapSection (get_runtime_object.Invoke (section, new object [0]));
+ AddSectionToCache (GetSectionCacheKey (sectionName, path), value);
+ return value;
+#endif
}
- static MethodInfo get_runtime_object = typeof (ConfigurationSection).GetMethod ("GetRuntimeObject", BindingFlags.NonPublic | BindingFlags.Instance);
+ static string GetCurrentPath (HttpContext ctx)
+ {
+ return (ctx != null && ctx.Request != null) ? ctx.Request.Path : HttpRuntime.AppDomainAppVirtualPath;
+ }
+
+ internal static void RemoveConfigurationFromCache (HttpContext ctx)
+ {
+ configurations.Remove (GetCurrentPath (ctx));
+ }
+
+ readonly static MethodInfo get_runtime_object = typeof (ConfigurationSection).GetMethod ("GetRuntimeObject", BindingFlags.NonPublic | BindingFlags.Instance);
public static object GetWebApplicationSection (string sectionName)
{
- _Configuration config = GetWebApplicationConfiguration ();
- if (config == null)
- return null;
+ string path = (HttpContext.Current == null
+ || HttpContext.Current.Request == null
+ || HttpContext.Current.Request.ApplicationPath == null
+ || HttpContext.Current.Request.ApplicationPath == "") ?
+ String.Empty : HttpContext.Current.Request.ApplicationPath;
- ConfigurationSection section = config.GetSection (sectionName);
- if (section == null)
- return null;
-
- return get_runtime_object.Invoke (section, new object [0]);
+ return GetSection (sectionName, path);
}
public static NameValueCollection AppSettings {
internal static IInternalConfigConfigurationFactory ConfigurationFactory {
get { return configFactory; }
}
-
- static string GetBasePath (string path)
- {
- if (path == "/" || path == "")
- return path;
-
- /* first if we can, map it to a physical path
- * to see if it corresponds to a file */
- if (HttpContext.Current != null
- && HttpContext.Current.Request != null) {
- string pd = HttpContext.Current.Request.MapPath (path);
-
- if (!Directory.Exists (pd)) {
- /* if it does, remove the file from the url */
- int i = path.LastIndexOf ('/');
- path = path.Substring (0, i);
- }
- }
-
- if (path.Length == 0)
- return path;
- /* remove excess /'s from the end of the virtual path */
- while (path [path.Length - 1] == '/')
- path = path.Substring (0, path.Length - 1);
+ static void AddSectionToCache (string key, object section)
+ {
+ if (sectionCache [key] != null)
+ return;
+
+ Hashtable tmpTable = (Hashtable) sectionCache.Clone ();
+ if (tmpTable.Contains (key))
+ return;
- return path;
+ tmpTable.Add (key, section);
+ sectionCache = tmpTable;
}
+ static string GetSectionCacheKey (string sectionName, string path)
+ {
+ return string.Concat (path, "/", sectionName);
+ }
+
#region stuff copied from WebConfigurationSettings
#if TARGET_J2EE
static internal IConfigurationSystem oldConfig {
#else
static internal IConfigurationSystem oldConfig;
static Web20DefaultConfig config;
- static IInternalConfigSystem configSystem;
+ //static IInternalConfigSystem configSystem;
#endif
const BindingFlags privStatic = BindingFlags.NonPublic | BindingFlags.Static;
static readonly object lockobj = new object ();
object [] args = new object [] {system};
changeConfig.Invoke (null, args);
- configSystem = system;
+ //configSystem = system;
}
}
}