|
//------------------------------------------------------------
// Copyright (c) Microsoft Corporation. All rights reserved.
//------------------------------------------------------------
#pragma warning disable 1634, 1691
namespace System.ServiceModel.Dispatcher
{
using System;
using System.Collections;
using System.ComponentModel;
using System.Diagnostics;
using System.Diagnostics.CodeAnalysis;
using System.Globalization;
using System.Reflection;
using System.Runtime;
using System.Runtime.InteropServices;
using System.ServiceModel;
using System.Xml;
// Thread Safety: This class is thread safe
public class QueryStringConverter
{
Hashtable defaultSupportedQueryStringTypes;
// the cache does not have a quota since it is per endpoint and is
// bounded by the number of types in the contract at the endpoint
Hashtable typeConverterCache;
public QueryStringConverter()
{
this.defaultSupportedQueryStringTypes = new Hashtable();
this.defaultSupportedQueryStringTypes.Add(typeof(Byte), null);
this.defaultSupportedQueryStringTypes.Add(typeof(SByte), null);
this.defaultSupportedQueryStringTypes.Add(typeof(Int16), null);
this.defaultSupportedQueryStringTypes.Add(typeof(Int32), null);
this.defaultSupportedQueryStringTypes.Add(typeof(Int64), null);
this.defaultSupportedQueryStringTypes.Add(typeof(UInt16), null);
this.defaultSupportedQueryStringTypes.Add(typeof(UInt32), null);
this.defaultSupportedQueryStringTypes.Add(typeof(UInt64), null);
this.defaultSupportedQueryStringTypes.Add(typeof(Single), null);
this.defaultSupportedQueryStringTypes.Add(typeof(Double), null);
this.defaultSupportedQueryStringTypes.Add(typeof(Boolean), null);
this.defaultSupportedQueryStringTypes.Add(typeof(Char), null);
this.defaultSupportedQueryStringTypes.Add(typeof(Decimal), null);
this.defaultSupportedQueryStringTypes.Add(typeof(String), null);
this.defaultSupportedQueryStringTypes.Add(typeof(Object), null);
this.defaultSupportedQueryStringTypes.Add(typeof(DateTime), null);
this.defaultSupportedQueryStringTypes.Add(typeof(TimeSpan), null);
this.defaultSupportedQueryStringTypes.Add(typeof(byte[]), null);
this.defaultSupportedQueryStringTypes.Add(typeof(Guid), null);
this.defaultSupportedQueryStringTypes.Add(typeof(Uri), null);
this.defaultSupportedQueryStringTypes.Add(typeof(DateTimeOffset), null);
this.typeConverterCache = new Hashtable();
}
public virtual bool CanConvert(Type type)
{
if (this.defaultSupportedQueryStringTypes.ContainsKey(type))
{
return true;
}
// otherwise check if its an enum
if (typeof(Enum).IsAssignableFrom(type))
{
return true;
}
// check if there's a typeconverter defined on the type
return (GetStringConverter(type) != null);
}
public virtual object ConvertStringToValue(string parameter, Type parameterType)
{
if (parameterType == null)
{
throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgumentNull("parameterType");
}
switch (Type.GetTypeCode(parameterType))
{
case TypeCode.Byte:
return parameter == null ? default(Byte) : XmlConvert.ToByte(parameter);
case TypeCode.SByte:
return parameter == null ? default(SByte) : XmlConvert.ToSByte(parameter);
case TypeCode.Int16:
return parameter == null ? default(Int16) : XmlConvert.ToInt16(parameter);
case TypeCode.Int32:
{
if (typeof(Enum).IsAssignableFrom(parameterType))
{
return Enum.Parse(parameterType, parameter, true);
}
else
{
return parameter == null ? default(Int32) : XmlConvert.ToInt32(parameter);
}
}
case TypeCode.Int64:
return parameter == null ? default(Int64) : XmlConvert.ToInt64(parameter);
case TypeCode.UInt16:
return parameter == null ? default(UInt16) : XmlConvert.ToUInt16(parameter);
case TypeCode.UInt32:
return parameter == null ? default(UInt32) : XmlConvert.ToUInt32(parameter);
case TypeCode.UInt64:
return parameter == null ? default(UInt64) : XmlConvert.ToUInt64(parameter);
case TypeCode.Single:
return parameter == null ? default(Single) : XmlConvert.ToSingle(parameter);
case TypeCode.Double:
return parameter == null ? default(Double) : XmlConvert.ToDouble(parameter);
case TypeCode.Char:
return parameter == null ? default(Char) : XmlConvert.ToChar(parameter);
case TypeCode.Decimal:
return parameter == null ? default(Decimal) : XmlConvert.ToDecimal(parameter);
case TypeCode.Boolean:
return parameter == null ? default(Boolean) : Convert.ToBoolean(parameter, CultureInfo.InvariantCulture);
case TypeCode.String:
return parameter;
case TypeCode.DateTime:
return parameter == null ? default(DateTime) : DateTime.Parse(parameter, CultureInfo.InvariantCulture, DateTimeStyles.RoundtripKind);
default:
{
if (parameterType == typeof(TimeSpan))
{
// support the XML as well as default way of representing timespans
TimeSpan result;
if (!TimeSpan.TryParse(parameter, out result))
{
result = parameter == null ? default(TimeSpan) : XmlConvert.ToTimeSpan(parameter);
}
return result;
}
else if (parameterType == typeof(Guid))
{
return parameter == null ? default(Guid) : XmlConvert.ToGuid(parameter);
}
else if (parameterType == typeof(DateTimeOffset))
{
return (parameter == null) ? default(DateTimeOffset) : DateTimeOffset.Parse(parameter, CultureInfo.InvariantCulture, DateTimeStyles.RoundtripKind | DateTimeStyles.AllowWhiteSpaces);
}
else if (parameterType == typeof(byte[]))
{
return (!string.IsNullOrEmpty(parameter)) ? Convert.FromBase64String(parameter) : new byte[] { };
}
else if (parameterType == typeof(Uri))
{
return (!string.IsNullOrEmpty(parameter)) ? new Uri(parameter, UriKind.RelativeOrAbsolute) : null;
}
else if (parameterType == typeof(object))
{
return parameter;
}
else
{
TypeConverter stringConverter = GetStringConverter(parameterType);
if (stringConverter == null)
{
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new NotSupportedException(
SR2.GetString(
SR2.TypeNotSupportedByQueryStringConverter,
parameterType.ToString(), this.GetType().Name)));
}
return stringConverter.ConvertFromInvariantString(parameter);
}
}
}
}
public virtual string ConvertValueToString(object parameter, Type parameterType)
{
if (parameterType == null)
{
throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgumentNull("parameterType");
}
if (parameterType.IsValueType && parameter == null)
{
throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgumentNull("parameter");
}
switch (Type.GetTypeCode(parameterType))
{
case TypeCode.Byte:
return XmlConvert.ToString((Byte) parameter);
case TypeCode.SByte:
return XmlConvert.ToString((SByte) parameter);
case TypeCode.Int16:
return XmlConvert.ToString((Int16) parameter);
case TypeCode.Int32:
{
if (typeof(Enum).IsAssignableFrom(parameterType))
{
return Enum.Format(parameterType, parameter, "G");
}
else
{
return XmlConvert.ToString((int) parameter);
}
}
case TypeCode.Int64:
return XmlConvert.ToString((Int64) parameter);
case TypeCode.UInt16:
return XmlConvert.ToString((UInt16) parameter);
case TypeCode.UInt32:
return XmlConvert.ToString((uint) parameter);
case TypeCode.UInt64:
return XmlConvert.ToString((UInt64) parameter);
case TypeCode.Single:
return XmlConvert.ToString((Single) parameter);
case TypeCode.Double:
return XmlConvert.ToString((double) parameter);
case TypeCode.Char:
return XmlConvert.ToString((char) parameter);
case TypeCode.Decimal:
return XmlConvert.ToString((decimal) parameter);
case TypeCode.Boolean:
return XmlConvert.ToString((bool) parameter);
case TypeCode.String:
return (string) parameter;
case TypeCode.DateTime:
return XmlConvert.ToString((DateTime) parameter, XmlDateTimeSerializationMode.RoundtripKind);
default:
{
if (parameterType == typeof(TimeSpan))
{
return XmlConvert.ToString((TimeSpan) parameter);
}
else if (parameterType == typeof(Guid))
{
return XmlConvert.ToString((Guid) parameter);
}
else if (parameterType == typeof(DateTimeOffset))
{
return XmlConvert.ToString((DateTimeOffset) parameter);
}
else if (parameterType == typeof(byte[]))
{
return (parameter != null) ? Convert.ToBase64String((byte[]) parameter, Base64FormattingOptions.None) : null;
}
else if (parameterType == typeof(Uri) || parameterType == typeof(object))
{
// URI or object
return (parameter != null) ? Convert.ToString(parameter, CultureInfo.InvariantCulture) : null;
}
else
{
TypeConverter stringConverter = GetStringConverter(parameterType);
if (stringConverter == null)
{
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new NotSupportedException(
SR2.GetString(
SR2.TypeNotSupportedByQueryStringConverter,
parameterType.ToString(), this.GetType().Name)));
}
else
{
return stringConverter.ConvertToInvariantString(parameter);
}
}
}
}
}
// hash table is safe for multiple readers single writer
[SuppressMessage("Reliability", "Reliability104:CaughtAndHandledExceptionsRule", Justification = "The exception is traced in the finally clause")]
TypeConverter GetStringConverter(Type parameterType)
{
if (this.typeConverterCache.ContainsKey(parameterType))
{
return (TypeConverter) this.typeConverterCache[parameterType];
}
TypeConverterAttribute[] typeConverterAttrs = parameterType.GetCustomAttributes(typeof(TypeConverterAttribute), true) as TypeConverterAttribute[];
if (typeConverterAttrs != null)
{
foreach (TypeConverterAttribute converterAttr in typeConverterAttrs)
{
Type converterType = Type.GetType(converterAttr.ConverterTypeName, false, true);
if (converterType != null)
{
TypeConverter converter = null;
Exception handledException = null;
try
{
converter = (TypeConverter) Activator.CreateInstance(converterType);
}
catch (TargetInvocationException e)
{
handledException = e;
}
catch (MemberAccessException e)
{
handledException = e;
}
catch (TypeLoadException e)
{
handledException = e;
}
catch (COMException e)
{
handledException = e;
}
catch (InvalidComObjectException e)
{
handledException = e;
}
finally
{
if (handledException != null)
{
if (Fx.IsFatal(handledException))
{
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(handledException);
}
DiagnosticUtility.TraceHandledException(handledException, TraceEventType.Warning);
}
}
if (converter == null)
{
continue;
}
if (converter.CanConvertTo(typeof(string)) && converter.CanConvertFrom(typeof(string)))
{
this.typeConverterCache.Add(parameterType, converter);
return converter;
}
}
}
}
return null;
}
}
}
|