225 lines
7.9 KiB
C#
225 lines
7.9 KiB
C#
|
//
|
||
|
// JsonQueryStringConverter.cs
|
||
|
//
|
||
|
// Author:
|
||
|
// Atsushi Enomoto <atsushi@ximian.com>
|
||
|
//
|
||
|
// Copyright (C) 2008 Novell, Inc (http://www.novell.com)
|
||
|
// Copyright 2014 Xamarin Inc. (http://www.xamarin.com)
|
||
|
//
|
||
|
// 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.
|
||
|
//
|
||
|
using System;
|
||
|
using System.Globalization;
|
||
|
using System.IO;
|
||
|
using System.Runtime.Serialization.Json;
|
||
|
using System.ServiceModel;
|
||
|
using System.ServiceModel.Description;
|
||
|
using System.Text;
|
||
|
using System.Xml;
|
||
|
|
||
|
namespace System.ServiceModel.Dispatcher
|
||
|
{
|
||
|
public class JsonQueryStringConverter : QueryStringConverter
|
||
|
{
|
||
|
DataContractJsonSerializer serializer = new DataContractJsonSerializer (typeof (object));
|
||
|
|
||
|
internal string CustomWrapperName { get; set; }
|
||
|
|
||
|
public override bool CanConvert (Type type)
|
||
|
{
|
||
|
// almost copy from QueryStringConverter, except that DBNull and XmlQualifiedName are supported
|
||
|
switch (Type.GetTypeCode (type)) {
|
||
|
//case TypeCode.DBNull:
|
||
|
case TypeCode.Empty:
|
||
|
return false;
|
||
|
case TypeCode.Object:
|
||
|
if (type == typeof (TimeSpan))
|
||
|
return true;
|
||
|
if (type == typeof (DateTimeOffset))
|
||
|
return true;
|
||
|
if (type == typeof (Guid))
|
||
|
return true;
|
||
|
if (type == typeof (XmlQualifiedName))
|
||
|
return true;
|
||
|
if (type == typeof (object))
|
||
|
return true;
|
||
|
// if (type.GetCustomAttributes (typeof (TypeConverterAttribute), true).Length > 0)
|
||
|
// return true;
|
||
|
|
||
|
// FIXME: it should return false for things like List<OfPrivateType>.
|
||
|
return type.IsPublic || type.IsNestedPublic;
|
||
|
default:
|
||
|
return true;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public override object ConvertStringToValue (string parameter, Type parameterType)
|
||
|
{
|
||
|
if (parameterType == null)
|
||
|
throw new ArgumentNullException ("parameterType");
|
||
|
|
||
|
if (!CanConvert (parameterType))
|
||
|
throw new NotSupportedException (String.Format ("Conversion from the argument parameterType '{0}' is not supported", parameterType));
|
||
|
|
||
|
// In general .NET JSON parser is sloppy. It accepts
|
||
|
// such a string that is actually invalid in terms of
|
||
|
// the target type in JSON context.
|
||
|
|
||
|
switch (Type.GetTypeCode (parameterType)) {
|
||
|
case TypeCode.String:
|
||
|
// LAMESPEC LAMESPEC LAMESPEC: we cannot give "foo" as the string value input (even if they are escaped as %22!)
|
||
|
if (parameter == null)
|
||
|
return null;
|
||
|
if (parameter.Length > 1 && parameter [0] == '"' && parameter [parameter.Length - 1] == '"')
|
||
|
return parameter.Substring (1, parameter.Length - 2);
|
||
|
else if (parameter [0] != '"')
|
||
|
return parameter;
|
||
|
break;
|
||
|
case TypeCode.Char:
|
||
|
return parameter != null ? Char.Parse (parameter): default (char);
|
||
|
case TypeCode.SByte:
|
||
|
return parameter != null ? SByte.Parse (parameter, CultureInfo.InvariantCulture): default (sbyte);
|
||
|
case TypeCode.Byte:
|
||
|
return parameter != null ? Byte.Parse (parameter, CultureInfo.InvariantCulture): default (byte);
|
||
|
case TypeCode.Int16:
|
||
|
return parameter != null ? Int16.Parse (parameter, CultureInfo.InvariantCulture): default (short);
|
||
|
case TypeCode.Int32:
|
||
|
return parameter != null ? Int32.Parse (parameter, CultureInfo.InvariantCulture): default (int);
|
||
|
case TypeCode.Int64:
|
||
|
return parameter != null ? Int64.Parse (parameter, CultureInfo.InvariantCulture): default (long);
|
||
|
case TypeCode.UInt16:
|
||
|
return parameter != null ? UInt16.Parse (parameter, CultureInfo.InvariantCulture): default (ushort);
|
||
|
case TypeCode.UInt32:
|
||
|
return parameter != null ? UInt32.Parse (parameter, CultureInfo.InvariantCulture): default (uint);
|
||
|
case TypeCode.UInt64:
|
||
|
return parameter != null ? UInt64.Parse (parameter, CultureInfo.InvariantCulture): default (ulong);
|
||
|
case TypeCode.DateTime:
|
||
|
return parameter != null ? DateTime.Parse (parameter, CultureInfo.InvariantCulture): default (DateTime);
|
||
|
case TypeCode.Boolean:
|
||
|
return parameter != null ? Boolean.Parse (parameter): default (bool);
|
||
|
case TypeCode.Single:
|
||
|
return parameter != null ? Single.Parse (parameter, CultureInfo.InvariantCulture): default (float);
|
||
|
case TypeCode.Double:
|
||
|
return parameter != null ? Double.Parse (parameter, CultureInfo.InvariantCulture): default (double);
|
||
|
case TypeCode.Decimal:
|
||
|
return parameter != null ? Decimal.Parse (parameter, CultureInfo.InvariantCulture): default (decimal);
|
||
|
}
|
||
|
|
||
|
if (parameter == null)
|
||
|
return null;
|
||
|
|
||
|
|
||
|
DataContractJsonSerializer serializer =
|
||
|
new DataContractJsonSerializer (parameterType);
|
||
|
// hmm, it costs so silly though.
|
||
|
return serializer.ReadObject (new MemoryStream (Encoding.UTF8.GetBytes (parameter)));
|
||
|
}
|
||
|
|
||
|
bool IsKnownType (Type t)
|
||
|
{
|
||
|
switch (Type.GetTypeCode (t)) {
|
||
|
case TypeCode.Object:
|
||
|
if (t == typeof (Guid) ||
|
||
|
t == typeof (DBNull) ||
|
||
|
t == typeof (DateTimeOffset) ||
|
||
|
t == typeof (TimeSpan) ||
|
||
|
t == typeof (XmlQualifiedName))
|
||
|
return true;
|
||
|
return false;
|
||
|
default:
|
||
|
return true;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public override string ConvertValueToString (object parameter, Type parameterType)
|
||
|
{
|
||
|
if (parameterType == null)
|
||
|
throw new ArgumentNullException ("parameterType");
|
||
|
|
||
|
if (!CanConvert (parameterType))
|
||
|
throw new NotSupportedException (String.Format ("Conversion from the argument parameterType '{0}' is not supported", parameterType));
|
||
|
|
||
|
if (parameter == null)
|
||
|
return null;
|
||
|
|
||
|
if (parameter is DBNull)
|
||
|
return "{}";
|
||
|
|
||
|
parameterType = ToActualType (parameterType);
|
||
|
|
||
|
if (parameter is IConvertible)
|
||
|
parameter = ((IConvertible) parameter).ToType (parameterType, CultureInfo.InvariantCulture);
|
||
|
|
||
|
switch (Type.GetTypeCode (parameterType)) {
|
||
|
case TypeCode.String:
|
||
|
string s = parameter is IFormattable ?
|
||
|
((IFormattable) parameter).ToString (null, CultureInfo.InvariantCulture) :
|
||
|
parameter.ToString ();
|
||
|
StringBuilder sb = new StringBuilder (s);
|
||
|
sb.Replace ("\"", "\\\"");
|
||
|
sb.Insert (0, "\"");
|
||
|
sb.Append ('\"');
|
||
|
return sb.ToString ();
|
||
|
default:
|
||
|
if (parameterType == typeof (XmlQualifiedName)) {
|
||
|
var qname = (XmlQualifiedName) parameter;
|
||
|
return String.Concat ("\"", qname.Name, ":", qname.Namespace, "\"");
|
||
|
}
|
||
|
var f = (parameter as IFormattable);
|
||
|
return (f == null) ? parameter.ToString () : f.ToString (null, CultureInfo.InvariantCulture);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
Type ToActualType (Type t)
|
||
|
{
|
||
|
switch (Type.GetTypeCode (t)) {
|
||
|
case TypeCode.DBNull: // though DBNull.Value input is converted to "{}". This result is used for String input.
|
||
|
case TypeCode.Char:
|
||
|
case TypeCode.String:
|
||
|
return typeof (string);
|
||
|
case TypeCode.SByte:
|
||
|
case TypeCode.Int16:
|
||
|
case TypeCode.Int32:
|
||
|
case TypeCode.Int64:
|
||
|
// return typeof (long);
|
||
|
return typeof (decimal);
|
||
|
case TypeCode.Byte:
|
||
|
case TypeCode.UInt16:
|
||
|
case TypeCode.UInt32:
|
||
|
case TypeCode.UInt64:
|
||
|
// return typeof (ulong);
|
||
|
return typeof (decimal);
|
||
|
case TypeCode.DateTime:
|
||
|
case TypeCode.Boolean:
|
||
|
return t;
|
||
|
case TypeCode.Single:
|
||
|
case TypeCode.Double:
|
||
|
// return typeof (double);
|
||
|
return typeof (decimal);
|
||
|
case TypeCode.Decimal:
|
||
|
return typeof (decimal);
|
||
|
default:
|
||
|
return t;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|