320 lines
12 KiB
C#
320 lines
12 KiB
C#
using Unosquare.Swan.Reflection;
|
|
using System;
|
|
using Unosquare.Swan.Components;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using System.Reflection;
|
|
using Unosquare.Swan.Attributes;
|
|
|
|
namespace Unosquare.Swan.Formatters {
|
|
/// <summary>
|
|
/// A very simple, light-weight JSON library written by Mario
|
|
/// to teach Geo how things are done
|
|
///
|
|
/// This is an useful helper for small tasks but it doesn't represent a full-featured
|
|
/// serializer such as the beloved Json.NET.
|
|
/// </summary>
|
|
public static partial class Json {
|
|
#region Constants
|
|
|
|
internal const String AddMethodName = "Add";
|
|
|
|
private const Char OpenObjectChar = '{';
|
|
private const Char CloseObjectChar = '}';
|
|
|
|
private const Char OpenArrayChar = '[';
|
|
private const Char CloseArrayChar = ']';
|
|
|
|
private const Char FieldSeparatorChar = ',';
|
|
private const Char ValueSeparatorChar = ':';
|
|
|
|
private const Char StringEscapeChar = '\\';
|
|
private const Char StringQuotedChar = '"';
|
|
|
|
private const String EmptyObjectLiteral = "{ }";
|
|
private const String EmptyArrayLiteral = "[ ]";
|
|
private const String TrueLiteral = "true";
|
|
private const String FalseLiteral = "false";
|
|
private const String NullLiteral = "null";
|
|
|
|
#endregion
|
|
|
|
private static readonly PropertyTypeCache PropertyTypeCache = new PropertyTypeCache();
|
|
private static readonly FieldTypeCache FieldTypeCache = new FieldTypeCache();
|
|
private static readonly CollectionCacheRepository<String> IgnoredPropertiesCache = new CollectionCacheRepository<String>();
|
|
|
|
#region Public API
|
|
|
|
/// <summary>
|
|
/// Serializes the specified object into a JSON string.
|
|
/// </summary>
|
|
/// <param name="obj">The object.</param>
|
|
/// <param name="format">if set to <c>true</c> it formats and indents the output.</param>
|
|
/// <param name="typeSpecifier">The type specifier. Leave null or empty to avoid setting.</param>
|
|
/// <param name="includeNonPublic">if set to <c>true</c> non-public getters will be also read.</param>
|
|
/// <param name="includedNames">The included property names.</param>
|
|
/// <param name="excludedNames">The excluded property names.</param>
|
|
/// <returns>
|
|
/// A <see cref="System.String" /> that represents the current object.
|
|
/// </returns>
|
|
/// <example>
|
|
/// The following example describes how to serialize a simple object.
|
|
/// <code>
|
|
/// using Unosquare.Swan.Formatters;
|
|
///
|
|
/// class Example
|
|
/// {
|
|
/// static void Main()
|
|
/// {
|
|
/// var obj = new { One = "One", Two = "Two" };
|
|
///
|
|
/// var serial = Json.Serialize(obj); // {"One": "One","Two": "Two"}
|
|
/// }
|
|
/// }
|
|
/// </code>
|
|
/// The following example details how to serialize an object using the <see cref="JsonPropertyAttribute"/>.
|
|
/// <code>
|
|
/// using Unosquare.Swan.Attributes;
|
|
/// using Unosquare.Swan.Formatters;
|
|
///
|
|
/// class Example
|
|
/// {
|
|
/// class JsonPropertyExample
|
|
/// {
|
|
/// [JsonProperty("data")]
|
|
/// public string Data { get; set; }
|
|
///
|
|
/// [JsonProperty("ignoredData", true)]
|
|
/// public string IgnoredData { get; set; }
|
|
/// }
|
|
///
|
|
/// static void Main()
|
|
/// {
|
|
/// var obj = new JsonPropertyExample() { Data = "OK", IgnoredData = "OK" };
|
|
///
|
|
/// // {"data": "OK"}
|
|
/// var serializedObj = Json.Serialize(obj);
|
|
/// }
|
|
/// }
|
|
/// </code>
|
|
/// </example>
|
|
public static String Serialize(
|
|
Object obj,
|
|
Boolean format = false,
|
|
String typeSpecifier = null,
|
|
Boolean includeNonPublic = false,
|
|
String[] includedNames = null,
|
|
String[] excludedNames = null) => Serialize(obj, format, typeSpecifier, includeNonPublic, includedNames, excludedNames, null);
|
|
|
|
/// <summary>
|
|
/// Serializes the specified object into a JSON string.
|
|
/// </summary>
|
|
/// <param name="obj">The object.</param>
|
|
/// <param name="format">if set to <c>true</c> it formats and indents the output.</param>
|
|
/// <param name="typeSpecifier">The type specifier. Leave null or empty to avoid setting.</param>
|
|
/// <param name="includeNonPublic">if set to <c>true</c> non-public getters will be also read.</param>
|
|
/// <param name="includedNames">The included property names.</param>
|
|
/// <param name="excludedNames">The excluded property names.</param>
|
|
/// <param name="parentReferences">The parent references.</param>
|
|
/// <returns>
|
|
/// A <see cref="System.String" /> that represents the current object.
|
|
/// </returns>
|
|
public static String Serialize(
|
|
Object obj,
|
|
Boolean format,
|
|
String typeSpecifier,
|
|
Boolean includeNonPublic,
|
|
String[] includedNames,
|
|
String[] excludedNames,
|
|
List<WeakReference> parentReferences) {
|
|
if(obj != null && (obj is String || Definitions.AllBasicValueTypes.Contains(obj.GetType()))) {
|
|
return SerializePrimitiveValue(obj);
|
|
}
|
|
|
|
SerializerOptions options = new SerializerOptions(
|
|
format,
|
|
typeSpecifier,
|
|
includedNames,
|
|
GetExcludedNames(obj?.GetType(), excludedNames),
|
|
includeNonPublic,
|
|
parentReferences);
|
|
|
|
return Serializer.Serialize(obj, 0, options);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Serializes the specified object only including the specified property names.
|
|
/// </summary>
|
|
/// <param name="obj">The object.</param>
|
|
/// <param name="format">if set to <c>true</c> it formats and indents the output.</param>
|
|
/// <param name="includeNames">The include names.</param>
|
|
/// <returns>A <see cref="System.String" /> that represents the current object.</returns>
|
|
/// <example>
|
|
/// The following example shows how to serialize a simple object including the specified properties.
|
|
/// <code>
|
|
/// using Unosquare.Swan.Formatters;
|
|
///
|
|
/// class Example
|
|
/// {
|
|
/// static void Main()
|
|
/// {
|
|
/// // object to serialize
|
|
/// var obj = new { One = "One", Two = "Two", Three = "Three" };
|
|
///
|
|
/// // the included names
|
|
/// var includedNames = new[] { "Two", "Three" };
|
|
///
|
|
/// // serialize only the included names
|
|
/// var data = Json.SerializeOnly(basicObject, true, includedNames);
|
|
/// // {"Two": "Two","Three": "Three" }
|
|
/// }
|
|
/// }
|
|
/// </code>
|
|
/// </example>
|
|
public static String SerializeOnly(Object obj, Boolean format, params String[] includeNames) {
|
|
SerializerOptions options = new SerializerOptions(format, null, includeNames);
|
|
|
|
return Serializer.Serialize(obj, 0, options);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Serializes the specified object excluding the specified property names.
|
|
/// </summary>
|
|
/// <param name="obj">The object.</param>
|
|
/// <param name="format">if set to <c>true</c> it formats and indents the output.</param>
|
|
/// <param name="excludeNames">The exclude names.</param>
|
|
/// <returns>A <see cref="System.String" /> that represents the current object.</returns>
|
|
/// <example>
|
|
/// The following code shows how to serialize a simple object exluding the specified properties.
|
|
/// <code>
|
|
/// using Unosquare.Swan.Formatters;
|
|
///
|
|
/// class Example
|
|
/// {
|
|
/// static void Main()
|
|
/// {
|
|
/// // object to serialize
|
|
/// var obj = new { One = "One", Two = "Two", Three = "Three" };
|
|
///
|
|
/// // the excluded names
|
|
/// var excludeNames = new[] { "Two", "Three" };
|
|
///
|
|
/// // serialize excluding
|
|
/// var data = Json.SerializeExcluding(basicObject, false, includedNames);
|
|
/// // {"One": "One"}
|
|
/// }
|
|
/// }
|
|
/// </code>
|
|
/// </example>
|
|
public static String SerializeExcluding(Object obj, Boolean format, params String[] excludeNames) {
|
|
SerializerOptions options = new SerializerOptions(format, null, null, excludeNames);
|
|
|
|
return Serializer.Serialize(obj, 0, options);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Deserializes the specified json string as either a Dictionary[string, object] or as a List[object]
|
|
/// depending on the syntax of the JSON string.
|
|
/// </summary>
|
|
/// <param name="json">The json.</param>
|
|
/// <returns>Type of the current deserializes.</returns>
|
|
/// <example>
|
|
/// The following code shows how to deserialize a JSON string into a Dictionary.
|
|
/// <code>
|
|
/// using Unosquare.Swan.Formatters;
|
|
///
|
|
/// class Example
|
|
/// {
|
|
/// static void Main()
|
|
/// {
|
|
/// // json to deserialize
|
|
/// var basicJson = "{\"One\":\"One\",\"Two\":\"Two\",\"Three\":\"Three\"}";
|
|
///
|
|
/// // deserializes the specified json into a Dictionary<string, object>.
|
|
/// var data = Json.Deserialize(basicJson);
|
|
/// }
|
|
/// }
|
|
/// </code>
|
|
/// </example>
|
|
public static Object Deserialize(String json) => Deserializer.DeserializeInternal(json);
|
|
|
|
/// <summary>
|
|
/// Deserializes the specified json string and converts it to the specified object type.
|
|
/// Non-public constructors and property setters are ignored.
|
|
/// </summary>
|
|
/// <typeparam name="T">The type of object to deserialize.</typeparam>
|
|
/// <param name="json">The json.</param>
|
|
/// <returns>The deserialized specified type object.</returns>
|
|
/// <example>
|
|
/// The following code describes how to deserialize a JSON string into an object of type T.
|
|
/// <code>
|
|
/// using Unosquare.Swan.Formatters;
|
|
///
|
|
/// class Example
|
|
/// {
|
|
/// static void Main()
|
|
/// {
|
|
/// // json type BasicJson to serialize
|
|
/// var basicJson = "{\"One\":\"One\",\"Two\":\"Two\",\"Three\":\"Three\"}";
|
|
///
|
|
/// // deserializes the specified string in a new instance of the type BasicJson.
|
|
/// var data = Json.Deserialize<BasicJson>(basicJson);
|
|
/// }
|
|
/// }
|
|
/// </code>
|
|
/// </example>
|
|
public static T Deserialize<T>(String json) => (T)Deserialize(json, typeof(T));
|
|
|
|
/// <summary>
|
|
/// Deserializes the specified json string and converts it to the specified object type.
|
|
/// </summary>
|
|
/// <typeparam name="T">The type of object to deserialize.</typeparam>
|
|
/// <param name="json">The json.</param>
|
|
/// <param name="includeNonPublic">if set to true, it also uses the non-public constructors and property setters.</param>
|
|
/// <returns>The deserialized specified type object.</returns>
|
|
public static T Deserialize<T>(String json, Boolean includeNonPublic) => (T)Deserialize(json, typeof(T), includeNonPublic);
|
|
|
|
/// <summary>
|
|
/// Deserializes the specified json string and converts it to the specified object type.
|
|
/// </summary>
|
|
/// <param name="json">The json.</param>
|
|
/// <param name="resultType">Type of the result.</param>
|
|
/// <param name="includeNonPublic">if set to true, it also uses the non-public constructors and property setters.</param>
|
|
/// <returns>Type of the current conversion from json result.</returns>
|
|
public static Object Deserialize(String json, Type resultType, Boolean includeNonPublic = false)
|
|
=> Converter.FromJsonResult(Deserializer.DeserializeInternal(json), resultType, includeNonPublic);
|
|
|
|
#endregion
|
|
|
|
#region Private API
|
|
|
|
private static String[] GetExcludedNames(Type type, String[] excludedNames) {
|
|
if(type == null) {
|
|
return excludedNames;
|
|
}
|
|
|
|
IEnumerable<String> excludedByAttr = IgnoredPropertiesCache.Retrieve(type, t => t.GetProperties()
|
|
.Where(x => Runtime.AttributeCache.RetrieveOne<JsonPropertyAttribute>(x)?.Ignored == true)
|
|
.Select(x => x.Name));
|
|
|
|
return excludedByAttr?.Any() != true
|
|
? excludedNames
|
|
: excludedNames == null
|
|
? excludedByAttr.ToArray()
|
|
: excludedByAttr.Intersect(excludedNames).ToArray();
|
|
}
|
|
|
|
private static String SerializePrimitiveValue(Object obj) {
|
|
switch(obj) {
|
|
case String stringValue:
|
|
return stringValue;
|
|
case Boolean boolValue:
|
|
return boolValue ? TrueLiteral : FalseLiteral;
|
|
default:
|
|
return obj.ToString();
|
|
}
|
|
}
|
|
#endregion
|
|
}
|
|
}
|