e79aa3c0ed
Former-commit-id: a2155e9bd80020e49e72e86c44da02a8ac0e57a4
844 lines
26 KiB
C#
844 lines
26 KiB
C#
//
|
|
// CustomizableFileSettingsProvider.cs
|
|
//
|
|
// Authors:
|
|
// Noriaki Okimoto <seara@ojk.sppd.ne.jp>
|
|
// Atsushi Enomoto <atsushi@ximian.com>
|
|
//
|
|
// (C)2007 Noriaki Okimoto
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining
|
|
// a copy of this software and associated documentation files (the
|
|
// "Software"), to deal in the Software without restriction, including
|
|
// without limitation the rights to use, copy, modify, merge, publish,
|
|
// distribute, sublicense, and/or sell copies of the Software, and to
|
|
// permit persons to whom the Software is furnished to do so, subject to
|
|
// the following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be
|
|
// included in all copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
// EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
|
// NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
|
// LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
|
// OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
|
// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
//
|
|
|
|
#if CONFIGURATION_DEP
|
|
|
|
using System;
|
|
using System.Collections;
|
|
using System.Collections.Generic;
|
|
using System.Collections.Specialized;
|
|
using System.Configuration;
|
|
using System.Globalization;
|
|
using System.IO;
|
|
using System.Reflection;
|
|
using System.Security.Cryptography;
|
|
using System.Text;
|
|
using System.Xml;
|
|
|
|
namespace System.Configuration
|
|
{
|
|
// location to store user configuration settings.
|
|
internal enum UserConfigLocationOption : uint
|
|
{
|
|
Product = 0x20,
|
|
Product_VersionMajor = 0x21,
|
|
Product_VersionMinor = 0x22,
|
|
Product_VersionBuild = 0x24,
|
|
Product_VersionRevision = 0x28,
|
|
Company_Product = 0x30,
|
|
Company_Product_VersionMajor = 0x31,
|
|
Company_Product_VersionMinor = 0x32,
|
|
Company_Product_VersionBuild = 0x34,
|
|
Company_Product_VersionRevision = 0x38,
|
|
Evidence = 0x40,
|
|
Other = 0x8000
|
|
}
|
|
|
|
internal class CustomizableFileSettingsProvider : SettingsProvider, IApplicationSettingsProvider
|
|
{
|
|
// KLUDGE WARNING.
|
|
//
|
|
// This is used from within System.Web to allow mapping of the ExeConfigFilename to
|
|
// the correct Web.config for the current request. Otherwise web applications will
|
|
// not be able to access settings from Web.config. The type assigned to this
|
|
// variable must descend from the ConfigurationFileMap class and its
|
|
// MachineConfigFilename will be used to set the ExeConfigFilename.
|
|
//
|
|
// This is necessary to fix bug #491531
|
|
#pragma warning disable 649
|
|
private static Type webConfigurationFileMapType;
|
|
#pragma warning restore 649
|
|
|
|
private static string userRoamingPath = "";
|
|
private static string userLocalPath = "";
|
|
private static string userRoamingPathPrevVersion = "";
|
|
private static string userLocalPathPrevVersion = "";
|
|
private static string userRoamingName = "user.config";
|
|
private static string userLocalName = "user.config";
|
|
private static string userRoamingBasePath = "";
|
|
private static string userLocalBasePath = "";
|
|
private static string CompanyName = "";
|
|
private static string ProductName = "";
|
|
private static string ForceVersion = "";
|
|
private static string[] ProductVersion;
|
|
|
|
// whether to include parts in the folder name or not:
|
|
private static bool isVersionMajor = false; // 0x0001 major version
|
|
private static bool isVersionMinor = false; // 0x0002 minor version
|
|
private static bool isVersionBuild = false; // 0x0004 build version
|
|
private static bool isVersionRevision = false; // 0x0008 revision
|
|
private static bool isCompany = true; // 0x0010 corporate name
|
|
private static bool isProduct = true; // 0x0020 product name
|
|
private static bool isEvidence = false; // 0x0040 evidence hash
|
|
|
|
private static bool userDefine = false; // 0x8000 ignore all above and use user definition
|
|
|
|
private static UserConfigLocationOption userConfig = UserConfigLocationOption.Company_Product;
|
|
|
|
public override void Initialize (string name, NameValueCollection config)
|
|
{
|
|
base.Initialize (name, config);
|
|
}
|
|
|
|
// full path to roaming user.config
|
|
internal static string UserRoamingFullPath {
|
|
get { return Path.Combine (userRoamingPath, userRoamingName); }
|
|
}
|
|
|
|
// full path to local user.config
|
|
internal static string UserLocalFullPath {
|
|
get { return Path.Combine (userLocalPath, userLocalName); }
|
|
}
|
|
|
|
// previous full path to roaming user.config
|
|
public static string PrevUserRoamingFullPath {
|
|
get { return Path.Combine (userRoamingPathPrevVersion, userRoamingName); }
|
|
}
|
|
|
|
// previous full path to local user.config
|
|
public static string PrevUserLocalFullPath {
|
|
get { return Path.Combine (userLocalPathPrevVersion, userLocalName); }
|
|
}
|
|
|
|
// path to roaming user.config
|
|
public static string UserRoamingPath {
|
|
get { return userRoamingPath; }
|
|
}
|
|
|
|
// path to local user.config
|
|
public static string UserLocalPath {
|
|
get { return userLocalPath; }
|
|
}
|
|
|
|
// file name which is equivalent to user.config, for roaming user
|
|
public static string UserRoamingName {
|
|
get { return userRoamingName; }
|
|
}
|
|
|
|
// file name which is equivalent to user.config, for local user
|
|
public static string UserLocalName {
|
|
get { return userLocalName; }
|
|
}
|
|
|
|
public static UserConfigLocationOption UserConfigSelector
|
|
{
|
|
get { return userConfig; }
|
|
set {
|
|
userConfig = value;
|
|
|
|
if (((uint) userConfig & 0x8000) != 0) {
|
|
isVersionMajor = false;
|
|
isVersionMinor = false;
|
|
isVersionBuild = false;
|
|
isVersionRevision = false;
|
|
isCompany = false;
|
|
return;
|
|
}
|
|
|
|
isVersionRevision = ((uint) userConfig & 0x0008) != 0;
|
|
isVersionBuild = isVersionRevision | ((uint)userConfig & 0x0004) != 0;
|
|
isVersionMinor = isVersionBuild | ((uint)userConfig & 0x0002) != 0;
|
|
isVersionMajor = IsVersionMinor | ((uint)userConfig & 0x0001) != 0;
|
|
|
|
isCompany = ((uint) userConfig & 0x0010) != 0;
|
|
isProduct = ((uint) userConfig & 0x0020) != 0;
|
|
}
|
|
}
|
|
|
|
// whether the path to include the major version.
|
|
public static bool IsVersionMajor
|
|
{
|
|
get { return isVersionMajor; }
|
|
set
|
|
{
|
|
isVersionMajor = value;
|
|
isVersionMinor = false;
|
|
isVersionBuild = false;
|
|
isVersionRevision = false;
|
|
}
|
|
}
|
|
|
|
// whether the path to include minor version.
|
|
public static bool IsVersionMinor
|
|
{
|
|
get { return isVersionMinor; }
|
|
set
|
|
{
|
|
isVersionMinor = value;
|
|
if (isVersionMinor)
|
|
isVersionMajor = true;
|
|
isVersionBuild = false;
|
|
isVersionRevision = false;
|
|
}
|
|
}
|
|
|
|
// whether the path to include build version.
|
|
public static bool IsVersionBuild
|
|
{
|
|
get { return isVersionBuild; }
|
|
set
|
|
{
|
|
isVersionBuild = value;
|
|
if (isVersionBuild) {
|
|
isVersionMajor = true;
|
|
isVersionMinor = true;
|
|
}
|
|
isVersionRevision = false;
|
|
}
|
|
}
|
|
|
|
// whether the path to include revision.
|
|
public static bool IsVersionRevision
|
|
{
|
|
get { return isVersionRevision; }
|
|
set
|
|
{
|
|
isVersionRevision = value;
|
|
if (isVersionRevision) {
|
|
isVersionMajor = true;
|
|
isVersionMinor = true;
|
|
isVersionBuild = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
// whether the path to include company name.
|
|
public static bool IsCompany
|
|
{
|
|
get { return isCompany; }
|
|
set { isCompany = value; }
|
|
}
|
|
|
|
// whether the path to include evidence hash.
|
|
public static bool IsEvidence
|
|
{
|
|
get { return isEvidence; }
|
|
set { isEvidence = value; }
|
|
}
|
|
|
|
// AssemblyCompanyAttribute->Namespace->"Program"
|
|
private static string GetCompanyName ()
|
|
{
|
|
Assembly assembly = Assembly.GetEntryAssembly ();
|
|
if (assembly == null)
|
|
assembly = Assembly.GetCallingAssembly ();
|
|
|
|
AssemblyCompanyAttribute [] attrs = (AssemblyCompanyAttribute []) assembly.GetCustomAttributes (typeof (AssemblyCompanyAttribute), true);
|
|
|
|
if ((attrs != null) && attrs.Length > 0) {
|
|
return attrs [0].Company;
|
|
}
|
|
|
|
MethodInfo entryPoint = assembly.EntryPoint;
|
|
Type entryType = entryPoint != null ? entryPoint.DeclaringType : null;
|
|
if (entryType != null && !String.IsNullOrEmpty (entryType.Namespace)) {
|
|
int end = entryType.Namespace.IndexOf ('.');
|
|
return end < 0 ? entryType.Namespace : entryType.Namespace.Substring (0, end);
|
|
}
|
|
return "Program";
|
|
}
|
|
|
|
private static string GetProductName ()
|
|
{
|
|
Assembly assembly = Assembly.GetEntryAssembly ();
|
|
if (assembly == null)
|
|
assembly = Assembly.GetCallingAssembly ();
|
|
|
|
byte [] pkt = assembly.GetName ().GetPublicKeyToken ();
|
|
return String.Format ("{0}_{1}_{2}",
|
|
AppDomain.CurrentDomain.FriendlyName,
|
|
pkt != null && pkt.Length > 0 ? "StrongName" : "Url",
|
|
GetEvidenceHash());
|
|
}
|
|
|
|
// Note: Changed from base64() to hex output to avoid unexpected chars like '\' or '/' with filesystem meaning.
|
|
// Otherwise eventually filenames, which are invalid on linux or windows, might be created.
|
|
// Signed-off-by: Carsten Schlote <schlote@vahanus.net>
|
|
// TODO: Compare with .NET. It might be also, that their way isn't suitable for Unix OS derivates (slahes in output)
|
|
private static string GetEvidenceHash ()
|
|
{
|
|
Assembly assembly = Assembly.GetEntryAssembly ();
|
|
if (assembly == null)
|
|
assembly = Assembly.GetCallingAssembly ();
|
|
|
|
byte [] pkt = assembly.GetName ().GetPublicKeyToken ();
|
|
byte [] hash = SHA1.Create ().ComputeHash (pkt != null && pkt.Length >0 ? pkt : Encoding.UTF8.GetBytes (assembly.EscapedCodeBase));
|
|
System.Text.StringBuilder evidence_string = new System.Text.StringBuilder();
|
|
foreach (byte b in hash)
|
|
evidence_string.AppendFormat("{0:x2}",b);
|
|
return evidence_string.ToString ();
|
|
}
|
|
|
|
private static string GetProductVersion ()
|
|
{
|
|
Assembly assembly = Assembly.GetEntryAssembly ();
|
|
if (assembly == null)
|
|
assembly = Assembly.GetCallingAssembly ();
|
|
if (assembly == null)
|
|
return string.Empty;
|
|
|
|
return assembly.GetName ().Version.ToString ();
|
|
}
|
|
|
|
private static void CreateUserConfigPath ()
|
|
{
|
|
if (userDefine)
|
|
return;
|
|
|
|
if (ProductName == "")
|
|
ProductName = GetProductName ();
|
|
if (CompanyName == "")
|
|
CompanyName = GetCompanyName ();
|
|
if (ForceVersion == "")
|
|
ProductVersion = GetProductVersion ().Split('.');
|
|
|
|
// C:\Documents and Settings\(user)\Application Data
|
|
if (userRoamingBasePath == "")
|
|
userRoamingPath = Environment.GetFolderPath (Environment.SpecialFolder.ApplicationData);
|
|
else
|
|
userRoamingPath = userRoamingBasePath;
|
|
|
|
// C:\Documents and Settings\(user)\Local Settings\Application Data (on Windows)
|
|
if (userLocalBasePath == "")
|
|
userLocalPath = Environment.GetFolderPath (Environment.SpecialFolder.LocalApplicationData);
|
|
else
|
|
userLocalPath = userLocalBasePath;
|
|
|
|
if (isCompany) {
|
|
userRoamingPath = Path.Combine (userRoamingPath, CompanyName);
|
|
userLocalPath = Path.Combine (userLocalPath, CompanyName);
|
|
}
|
|
|
|
if (isProduct) {
|
|
if (isEvidence) {
|
|
Assembly assembly = Assembly.GetEntryAssembly ();
|
|
if (assembly == null)
|
|
assembly = Assembly.GetCallingAssembly ();
|
|
byte [] pkt = assembly.GetName ().GetPublicKeyToken ();
|
|
ProductName = String.Format ("{0}_{1}_{2}",
|
|
ProductName,
|
|
pkt != null ? "StrongName" : "Url",
|
|
GetEvidenceHash());
|
|
}
|
|
userRoamingPath = Path.Combine (userRoamingPath, ProductName);
|
|
userLocalPath = Path.Combine (userLocalPath, ProductName);
|
|
|
|
}
|
|
|
|
string versionName;
|
|
|
|
if (ForceVersion == "") {
|
|
if (isVersionRevision)
|
|
versionName = String.Format ("{0}.{1}.{2}.{3}", ProductVersion [0], ProductVersion [1], ProductVersion [2], ProductVersion [3]);
|
|
else if (isVersionBuild)
|
|
versionName = String.Format ("{0}.{1}.{2}", ProductVersion [0], ProductVersion [1], ProductVersion [2]);
|
|
else if (isVersionMinor)
|
|
versionName = String.Format ("{0}.{1}", ProductVersion [0], ProductVersion [1]);
|
|
else if (isVersionMajor)
|
|
versionName = ProductVersion [0];
|
|
else
|
|
versionName = "";
|
|
}
|
|
else
|
|
versionName = ForceVersion;
|
|
|
|
string prevVersionRoaming = PrevVersionPath (userRoamingPath, versionName);
|
|
string prevVersionLocal = PrevVersionPath (userLocalPath, versionName);
|
|
|
|
userRoamingPath = Path.Combine (userRoamingPath, versionName);
|
|
userLocalPath = Path.Combine (userLocalPath, versionName);
|
|
if (prevVersionRoaming != "")
|
|
userRoamingPathPrevVersion = Path.Combine(userRoamingPath, prevVersionRoaming);
|
|
if (prevVersionLocal != "")
|
|
userLocalPathPrevVersion = Path.Combine(userLocalPath, prevVersionLocal);
|
|
}
|
|
|
|
// string for the previous version. It ignores newer ones.
|
|
private static string PrevVersionPath (string dirName, string currentVersion)
|
|
{
|
|
string prevVersionString = "";
|
|
|
|
if (!Directory.Exists(dirName))
|
|
return prevVersionString;
|
|
DirectoryInfo currentDir = new DirectoryInfo (dirName);
|
|
foreach (DirectoryInfo dirInfo in currentDir.GetDirectories ())
|
|
if (String.Compare (currentVersion, dirInfo.Name, StringComparison.Ordinal) > 0)
|
|
if (String.Compare (prevVersionString, dirInfo.Name, StringComparison.Ordinal) < 0)
|
|
prevVersionString = dirInfo.Name;
|
|
|
|
return prevVersionString;
|
|
}
|
|
|
|
// sets the explicit path to store roaming user.config or equivalent.
|
|
// (returns the path validity.)
|
|
public static bool SetUserRoamingPath (string configPath)
|
|
{
|
|
if (CheckPath (configPath))
|
|
{
|
|
userRoamingBasePath = configPath;
|
|
return true;
|
|
}
|
|
else
|
|
return false;
|
|
}
|
|
|
|
// sets the explicit path to store local user.config or equivalent.
|
|
// (returns the path validity.)
|
|
public static bool SetUserLocalPath (string configPath)
|
|
{
|
|
if (CheckPath (configPath))
|
|
{
|
|
userLocalBasePath = configPath;
|
|
return true;
|
|
}
|
|
else
|
|
return false;
|
|
}
|
|
|
|
private static bool CheckFileName (string configFile)
|
|
{
|
|
/*
|
|
char[] invalidFileChars = Path.GetInvalidFileNameChars();
|
|
|
|
foreach (char invalidChar in invalidFileChars)
|
|
{
|
|
if (configFile.Contains(invalidChar.ToString()))
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
*/
|
|
return configFile.IndexOfAny (Path.GetInvalidFileNameChars ()) < 0;
|
|
}
|
|
|
|
// sets the explicit roaming file name which is user.config equivalent.
|
|
// (returns the file name validity.)
|
|
public static bool SetUserRoamingFileName (string configFile)
|
|
{
|
|
if (CheckFileName (configFile))
|
|
{
|
|
userRoamingName = configFile;
|
|
return true;
|
|
}
|
|
else
|
|
return false;
|
|
}
|
|
|
|
// sets the explicit local file name which is user.config equivalent.
|
|
// (returns the file name validity.)
|
|
public static bool SetUserLocalFileName (string configFile)
|
|
{
|
|
if (CheckFileName (configFile))
|
|
{
|
|
userLocalName = configFile;
|
|
return true;
|
|
}
|
|
else
|
|
return false;
|
|
}
|
|
|
|
// sets the explicit company name for folder.
|
|
// (returns the file name validity.)
|
|
public static bool SetCompanyName (string companyName)
|
|
{
|
|
if (CheckFileName (companyName))
|
|
{
|
|
CompanyName = companyName;
|
|
return true;
|
|
}
|
|
else
|
|
return false;
|
|
}
|
|
|
|
// sets the explicit product name for folder.
|
|
// (returns the file name validity.)
|
|
public static bool SetProductName (string productName)
|
|
{
|
|
if (CheckFileName (productName))
|
|
{
|
|
ProductName = productName;
|
|
return true;
|
|
}
|
|
else
|
|
return false;
|
|
}
|
|
|
|
// sets the explicit major version for folder.
|
|
public static bool SetVersion (int major)
|
|
{
|
|
ForceVersion = string.Format ("{0}", major);
|
|
return true;
|
|
}
|
|
|
|
// sets the explicit major and minor versions for folder.
|
|
public static bool SetVersion (int major, int minor)
|
|
{
|
|
ForceVersion = string.Format ("{0}.{1}", major, minor);
|
|
return true;
|
|
}
|
|
|
|
// sets the explicit major/minor/build numbers for folder.
|
|
public static bool SetVersion (int major, int minor, int build)
|
|
{
|
|
ForceVersion = string.Format ("{0}.{1}.{2}", major, minor, build);
|
|
return true;
|
|
}
|
|
|
|
// sets the explicit major/minor/build/revision numbers for folder.
|
|
public static bool SetVersion (int major, int minor, int build, int revision)
|
|
{
|
|
ForceVersion = string.Format ("{0}.{1}.{2}.{3}", major, minor, build, revision);
|
|
return true;
|
|
}
|
|
|
|
// sets the explicit version number string for folder.
|
|
public static bool SetVersion (string forceVersion)
|
|
{
|
|
if (CheckFileName (forceVersion))
|
|
{
|
|
ForceVersion = forceVersion;
|
|
return true;
|
|
}
|
|
else
|
|
return false;
|
|
}
|
|
|
|
private static bool CheckPath (string configPath)
|
|
{
|
|
char[] invalidPathChars = Path.GetInvalidPathChars ();
|
|
|
|
/*
|
|
foreach (char invalidChar in invalidPathChars)
|
|
{
|
|
if (configPath.Contains (invalidChar.ToString()))
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
*/
|
|
if (configPath.IndexOfAny (invalidPathChars) >= 0)
|
|
return false;
|
|
|
|
string folder = configPath;
|
|
string fileName;
|
|
while ((fileName = Path.GetFileName (folder)) != "")
|
|
{
|
|
if (!CheckFileName (fileName))
|
|
{
|
|
return false;
|
|
}
|
|
folder = Path.GetDirectoryName (folder);
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
public override string Name {
|
|
get { return base.Name; }
|
|
}
|
|
|
|
string app_name = String.Empty;//"OJK.CustomSetting.CustomizableLocalFileSettingsProvider";
|
|
public override string ApplicationName {
|
|
get { return app_name; }
|
|
set { app_name = value; }
|
|
}
|
|
|
|
private ExeConfigurationFileMap exeMapCurrent = null;
|
|
private ExeConfigurationFileMap exeMapPrev = null;
|
|
private SettingsPropertyValueCollection values = null;
|
|
|
|
/// <remarks>
|
|
/// Hack to remove the XmlDeclaration that the XmlSerializer adds.
|
|
/// <br />
|
|
/// see <a href="https://github.com/mono/mono/pull/2273">Issue 2273</a> for details
|
|
/// </remarks>
|
|
private string StripXmlHeader (string serializedValue)
|
|
{
|
|
if (serializedValue == null)
|
|
{
|
|
return string.Empty;
|
|
}
|
|
|
|
XmlDocument doc = new XmlDocument ();
|
|
XmlElement valueXml = doc.CreateElement ("value");
|
|
valueXml.InnerXml = serializedValue;
|
|
|
|
foreach (XmlNode child in valueXml.ChildNodes) {
|
|
if (child.NodeType == XmlNodeType.XmlDeclaration) {
|
|
valueXml.RemoveChild (child);
|
|
break;
|
|
}
|
|
}
|
|
|
|
// InnerXml will give you well-formed XML that you could save as a separate document, and
|
|
// InnerText will immediately give you a pure-text representation of this inner XML.
|
|
return valueXml.InnerXml;
|
|
}
|
|
|
|
private void SaveProperties (ExeConfigurationFileMap exeMap, SettingsPropertyValueCollection collection, ConfigurationUserLevel level, SettingsContext context, bool checkUserLevel)
|
|
{
|
|
Configuration config = ConfigurationManager.OpenMappedExeConfiguration (exeMap, level);
|
|
|
|
UserSettingsGroup userGroup = config.GetSectionGroup ("userSettings") as UserSettingsGroup;
|
|
bool isRoaming = (level == ConfigurationUserLevel.PerUserRoaming);
|
|
|
|
if (userGroup == null) {
|
|
userGroup = new UserSettingsGroup ();
|
|
config.SectionGroups.Add ("userSettings", userGroup);
|
|
}
|
|
ApplicationSettingsBase asb = context.CurrentSettings;
|
|
string class_name = NormalizeInvalidXmlChars ((asb != null ? asb.GetType () : typeof (ApplicationSettingsBase)).FullName);
|
|
ClientSettingsSection userSection = null;
|
|
ConfigurationSection cnf = userGroup.Sections.Get (class_name);
|
|
userSection = cnf as ClientSettingsSection;
|
|
if (userSection == null) {
|
|
userSection = new ClientSettingsSection ();
|
|
userGroup.Sections.Add (class_name, userSection);
|
|
}
|
|
|
|
bool hasChanges = false;
|
|
|
|
if (userSection == null)
|
|
return;
|
|
|
|
foreach (SettingsPropertyValue value in collection) {
|
|
if (checkUserLevel && value.Property.Attributes.Contains (typeof (SettingsManageabilityAttribute)) != isRoaming)
|
|
continue;
|
|
// The default impl does not save the ApplicationScopedSetting properties
|
|
if (value.Property.Attributes.Contains (typeof (ApplicationScopedSettingAttribute)))
|
|
continue;
|
|
|
|
hasChanges = true;
|
|
SettingElement element = userSection.Settings.Get (value.Name);
|
|
if (element == null) {
|
|
element = new SettingElement (value.Name, value.Property.SerializeAs);
|
|
userSection.Settings.Add (element);
|
|
}
|
|
if (element.Value.ValueXml == null)
|
|
element.Value.ValueXml = new XmlDocument ().CreateElement ("value");
|
|
switch (value.Property.SerializeAs) {
|
|
case SettingsSerializeAs.Xml:
|
|
element.Value.ValueXml.InnerXml = StripXmlHeader (value.SerializedValue as string);
|
|
break;
|
|
case SettingsSerializeAs.String:
|
|
element.Value.ValueXml.InnerText = value.SerializedValue as string;
|
|
break;
|
|
case SettingsSerializeAs.Binary:
|
|
element.Value.ValueXml.InnerText = value.SerializedValue != null ? Convert.ToBase64String (value.SerializedValue as byte []) : string.Empty;
|
|
break;
|
|
default:
|
|
throw new NotImplementedException ();
|
|
}
|
|
}
|
|
if (hasChanges)
|
|
config.Save (ConfigurationSaveMode.Minimal, true);
|
|
}
|
|
|
|
// NOTE: We should add here all the chars that are valid in a name of a class (Ecma-wise),
|
|
// but invalid in an xml element name, and provide a better impl if we get too many of them.
|
|
string NormalizeInvalidXmlChars (string str)
|
|
{
|
|
char [] invalid_chars = new char [] { '+' };
|
|
|
|
if (str == null || str.IndexOfAny (invalid_chars) == -1)
|
|
return str;
|
|
|
|
// Replace with its hexadecimal values.
|
|
str = str.Replace ("+", "_x002B_");
|
|
return str;
|
|
}
|
|
|
|
private void LoadPropertyValue (SettingsPropertyCollection collection, SettingElement element, bool allowOverwrite)
|
|
{
|
|
SettingsProperty prop = collection [element.Name];
|
|
if (prop == null) { // see bug #343459
|
|
prop = new SettingsProperty (element.Name);
|
|
collection.Add (prop);
|
|
}
|
|
|
|
SettingsPropertyValue value = new SettingsPropertyValue (prop);
|
|
value.IsDirty = false;
|
|
if (element.Value.ValueXml != null) {
|
|
switch (value.Property.SerializeAs) {
|
|
case SettingsSerializeAs.Xml:
|
|
value.SerializedValue = element.Value.ValueXml.InnerXml;
|
|
break;
|
|
case SettingsSerializeAs.String:
|
|
value.SerializedValue = element.Value.ValueXml.InnerText.Trim ();
|
|
break;
|
|
case SettingsSerializeAs.Binary:
|
|
value.SerializedValue = Convert.FromBase64String (element.Value.ValueXml.InnerText);
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
value.SerializedValue = prop.DefaultValue;
|
|
try
|
|
{
|
|
if (allowOverwrite)
|
|
values.Remove (element.Name);
|
|
values.Add (value);
|
|
} catch (ArgumentException ex) {
|
|
throw new ConfigurationErrorsException (string.Format (
|
|
CultureInfo.InvariantCulture,
|
|
"Failed to load value for '{0}'.",
|
|
element.Name), ex);
|
|
}
|
|
}
|
|
|
|
private void LoadProperties (ExeConfigurationFileMap exeMap, SettingsPropertyCollection collection, ConfigurationUserLevel level, string sectionGroupName, bool allowOverwrite, string groupName)
|
|
{
|
|
Configuration config = ConfigurationManager.OpenMappedExeConfiguration (exeMap,level);
|
|
|
|
ConfigurationSectionGroup sectionGroup = config.GetSectionGroup (sectionGroupName);
|
|
if (sectionGroup != null) {
|
|
foreach (ConfigurationSection configSection in sectionGroup.Sections) {
|
|
if (configSection.SectionInformation.Name != groupName)
|
|
continue;
|
|
|
|
ClientSettingsSection clientSection = configSection as ClientSettingsSection;
|
|
if (clientSection == null)
|
|
continue;
|
|
|
|
foreach (SettingElement element in clientSection.Settings) {
|
|
LoadPropertyValue(collection, element, allowOverwrite);
|
|
}
|
|
// Only the first one seems to be processed by MS
|
|
break;
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
public override void SetPropertyValues (SettingsContext context, SettingsPropertyValueCollection collection)
|
|
{
|
|
CreateExeMap ();
|
|
|
|
if (UserLocalFullPath == UserRoamingFullPath)
|
|
{
|
|
SaveProperties (exeMapCurrent, collection, ConfigurationUserLevel.PerUserRoaming, context, false);
|
|
} else {
|
|
SaveProperties (exeMapCurrent, collection, ConfigurationUserLevel.PerUserRoaming, context, true);
|
|
SaveProperties (exeMapCurrent, collection, ConfigurationUserLevel.PerUserRoamingAndLocal, context, true);
|
|
}
|
|
}
|
|
|
|
public override SettingsPropertyValueCollection GetPropertyValues (SettingsContext context, SettingsPropertyCollection collection)
|
|
{
|
|
CreateExeMap ();
|
|
|
|
values = new SettingsPropertyValueCollection ();
|
|
string groupName = context ["GroupName"] as string;
|
|
groupName = NormalizeInvalidXmlChars (groupName); // we likely saved the element removing the non valid xml chars.
|
|
LoadProperties (exeMapCurrent, collection, ConfigurationUserLevel.None, "applicationSettings", false, groupName);
|
|
LoadProperties (exeMapCurrent, collection, ConfigurationUserLevel.None, "userSettings", false, groupName);
|
|
|
|
LoadProperties (exeMapCurrent, collection, ConfigurationUserLevel.PerUserRoaming, "userSettings", true, groupName);
|
|
LoadProperties (exeMapCurrent, collection, ConfigurationUserLevel.PerUserRoamingAndLocal, "userSettings", true, groupName);
|
|
|
|
// create default values if not exist
|
|
foreach (SettingsProperty p in collection)
|
|
if (values [p.Name] == null)
|
|
values.Add (new SettingsPropertyValue (p));
|
|
return values;
|
|
}
|
|
|
|
/// creates an ExeConfigurationFileMap
|
|
private void CreateExeMap ()
|
|
{
|
|
if (exeMapCurrent == null) {
|
|
CreateUserConfigPath ();
|
|
|
|
// current version
|
|
exeMapCurrent = new ExeConfigurationFileMap ();
|
|
|
|
// exeMapCurrent.ExeConfigFilename = System.Windows.Forms.Application.ExecutablePath + ".config";
|
|
Assembly entry = Assembly.GetEntryAssembly () ?? Assembly.GetExecutingAssembly ();
|
|
exeMapCurrent.ExeConfigFilename = entry.Location + ".config";
|
|
exeMapCurrent.LocalUserConfigFilename = UserLocalFullPath;
|
|
exeMapCurrent.RoamingUserConfigFilename = UserRoamingFullPath;
|
|
|
|
if (webConfigurationFileMapType != null && typeof (ConfigurationFileMap).IsAssignableFrom (webConfigurationFileMapType)) {
|
|
try {
|
|
ConfigurationFileMap cfgFileMap = Activator.CreateInstance (webConfigurationFileMapType) as ConfigurationFileMap;
|
|
if (cfgFileMap != null) {
|
|
string fpath = cfgFileMap.MachineConfigFilename;
|
|
if (!String.IsNullOrEmpty (fpath))
|
|
exeMapCurrent.ExeConfigFilename = fpath;
|
|
}
|
|
} catch {
|
|
// ignore
|
|
}
|
|
}
|
|
|
|
// previous version
|
|
if ((PrevUserLocalFullPath != "") && (PrevUserRoamingFullPath != ""))
|
|
{
|
|
exeMapPrev = new ExeConfigurationFileMap();
|
|
// exeMapPrev.ExeConfigFilename = System.Windows.Forms.Application.ExecutablePath + ".config";
|
|
exeMapPrev.ExeConfigFilename = entry.Location + ".config";
|
|
exeMapPrev.LocalUserConfigFilename = PrevUserLocalFullPath;
|
|
exeMapPrev.RoamingUserConfigFilename = PrevUserRoamingFullPath;
|
|
}
|
|
}
|
|
}
|
|
|
|
// FIXME: implement
|
|
public SettingsPropertyValue GetPreviousVersion (SettingsContext context, SettingsProperty property)
|
|
{
|
|
return null;
|
|
}
|
|
|
|
public void Reset (SettingsContext context)
|
|
{
|
|
if (values != null) {
|
|
foreach (SettingsPropertyValue propertyValue in values) {
|
|
// Can't use propertyValue.Property.DefaultValue
|
|
// as it may cause InvalidCastException (see bug# 532180)
|
|
values[propertyValue.Name].PropertyValue = propertyValue.Reset ();
|
|
}
|
|
}
|
|
}
|
|
|
|
// FIXME: implement
|
|
public void Upgrade (SettingsContext context, SettingsPropertyCollection properties)
|
|
{
|
|
}
|
|
|
|
public static void setCreate ()
|
|
{
|
|
CreateUserConfigPath();
|
|
}
|
|
}
|
|
}
|
|
|
|
#endif
|