Files
UnrealEngineUWP/Engine/Source/Programs/DotNETCommon/DotNETUtilities/JsonObject.cs
Ben Marsh 13d012685f Merging copyright update from 4.19 branch.
#rb none
#rnx
#jira

[CL 3818977 by Ben Marsh in Staging-4.19 branch]
2018-01-02 15:30:26 -05:00

716 lines
20 KiB
C#

// Copyright 1998-2018 Epic Games, Inc. All Rights Reserved.
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Text;
namespace Tools.DotNETCommon
{
/// <summary>
/// Exception thrown for errors parsing JSON files
/// </summary>
public class JsonParseException : Exception
{
/// <summary>
/// Constructor
/// </summary>
/// <param name="Format">Format string</param>
/// <param name="Args">Optional arguments</param>
public JsonParseException(string Format, params object[] Args)
: base(String.Format(Format, Args))
{
}
}
/// <summary>
/// Stores a JSON object in memory
/// </summary>
public class JsonObject
{
Dictionary<string, object> RawObject;
/// <summary>
/// Construct a JSON object from the raw string -> object dictionary
/// </summary>
/// <param name="InRawObject">Raw object parsed from disk</param>
public JsonObject(Dictionary<string, object> InRawObject)
{
RawObject = new Dictionary<string, object>(InRawObject, StringComparer.InvariantCultureIgnoreCase);
}
/// <summary>
/// Read a JSON file from disk and construct a JsonObject from it
/// </summary>
/// <param name="File">File to read from</param>
/// <returns>New JsonObject instance</returns>
public static JsonObject Read(FileReference File)
{
string Text = FileReference.ReadAllText(File);
try
{
return Parse(Text);
}
catch(Exception Ex)
{
throw new JsonParseException("Unable to parse {0}: {1}", File, Ex.Message);
}
}
/// <summary>
/// Tries to read a JSON file from disk
/// </summary>
/// <param name="FileName">File to read from</param>
/// <param name="Result">On success, receives the parsed object</param>
/// <returns>True if the file was read, false otherwise</returns>
public static bool TryRead(FileReference FileName, out JsonObject Result)
{
if (!FileReference.Exists(FileName))
{
Result = null;
return false;
}
string Text = FileReference.ReadAllText(FileName);
return TryParse(Text, out Result);
}
/// <summary>
/// Parse a JsonObject from the given raw text string
/// </summary>
/// <param name="Text">The text to parse</param>
/// <returns>New JsonObject instance</returns>
public static JsonObject Parse(string Text)
{
Dictionary<string, object> CaseSensitiveRawObject = (Dictionary<string, object>)fastJSON.JSON.Instance.Parse(Text);
return new JsonObject(CaseSensitiveRawObject);
}
/// <summary>
/// Try to parse a JsonObject from the given raw text string
/// </summary>
/// <param name="Text">The text to parse</param>
/// <param name="Result">On success, receives the new JsonObject</param>
/// <returns>True if the object was parsed</returns>
public static bool TryParse(string Text, out JsonObject Result)
{
try
{
Result = Parse(Text);
return true;
}
catch (Exception)
{
Result = null;
return false;
}
}
/// <summary>
/// List of key names in this object
/// </summary>
public IEnumerable<string> KeyNames
{
get { return RawObject.Keys; }
}
/// <summary>
/// Gets a string field by the given name from the object, throwing an exception if it is not there or cannot be parsed.
/// </summary>
/// <param name="FieldName">Name of the field to get</param>
/// <returns>The field value</returns>
public string GetStringField(string FieldName)
{
string StringValue;
if (!TryGetStringField(FieldName, out StringValue))
{
throw new JsonParseException("Missing or invalid '{0}' field", FieldName);
}
return StringValue;
}
/// <summary>
/// Tries to read a string field by the given name from the object
/// </summary>
/// <param name="FieldName">Name of the field to get</param>
/// <param name="Result">On success, receives the field value</param>
/// <returns>True if the field could be read, false otherwise</returns>
public bool TryGetStringField(string FieldName, out string Result)
{
object RawValue;
if (RawObject.TryGetValue(FieldName, out RawValue) && (RawValue is string))
{
Result = (string)RawValue;
return true;
}
else
{
Result = null;
return false;
}
}
/// <summary>
/// Gets a string array field by the given name from the object, throwing an exception if it is not there or cannot be parsed.
/// </summary>
/// <param name="FieldName">Name of the field to get</param>
/// <returns>The field value</returns>
public string[] GetStringArrayField(string FieldName)
{
string[] StringValues;
if (!TryGetStringArrayField(FieldName, out StringValues))
{
throw new JsonParseException("Missing or invalid '{0}' field", FieldName);
}
return StringValues;
}
/// <summary>
/// Tries to read a string array field by the given name from the object
/// </summary>
/// <param name="FieldName">Name of the field to get</param>
/// <param name="Result">On success, receives the field value</param>
/// <returns>True if the field could be read, false otherwise</returns>
public bool TryGetStringArrayField(string FieldName, out string[] Result)
{
object RawValue;
if (RawObject.TryGetValue(FieldName, out RawValue) && (RawValue is IEnumerable<object>) && ((IEnumerable<object>)RawValue).All(x => x is string))
{
Result = ((IEnumerable<object>)RawValue).Select(x => (string)x).ToArray();
return true;
}
else
{
Result = null;
return false;
}
}
/// <summary>
/// Gets a boolean field by the given name from the object, throwing an exception if it is not there or cannot be parsed.
/// </summary>
/// <param name="FieldName">Name of the field to get</param>
/// <returns>The field value</returns>
public bool GetBoolField(string FieldName)
{
bool BoolValue;
if (!TryGetBoolField(FieldName, out BoolValue))
{
throw new JsonParseException("Missing or invalid '{0}' field", FieldName);
}
return BoolValue;
}
/// <summary>
/// Tries to read a bool field by the given name from the object
/// </summary>
/// <param name="FieldName">Name of the field to get</param>
/// <param name="Result">On success, receives the field value</param>
/// <returns>True if the field could be read, false otherwise</returns>
public bool TryGetBoolField(string FieldName, out bool Result)
{
object RawValue;
if (RawObject.TryGetValue(FieldName, out RawValue) && (RawValue is Boolean))
{
Result = (bool)RawValue;
return true;
}
else
{
Result = false;
return false;
}
}
/// <summary>
/// Gets an integer field by the given name from the object, throwing an exception if it is not there or cannot be parsed.
/// </summary>
/// <param name="FieldName">Name of the field to get</param>
/// <returns>The field value</returns>
public int GetIntegerField(string FieldName)
{
int IntegerValue;
if (!TryGetIntegerField(FieldName, out IntegerValue))
{
throw new JsonParseException("Missing or invalid '{0}' field", FieldName);
}
return IntegerValue;
}
/// <summary>
/// Tries to read an integer field by the given name from the object
/// </summary>
/// <param name="FieldName">Name of the field to get</param>
/// <param name="Result">On success, receives the field value</param>
/// <returns>True if the field could be read, false otherwise</returns>
public bool TryGetIntegerField(string FieldName, out int Result)
{
object RawValue;
if (!RawObject.TryGetValue(FieldName, out RawValue) || !int.TryParse(RawValue.ToString(), out Result))
{
Result = 0;
return false;
}
return true;
}
/// <summary>
/// Tries to read an unsigned integer field by the given name from the object
/// </summary>
/// <param name="FieldName">Name of the field to get</param>
/// <param name="Result">On success, receives the field value</param>
/// <returns>True if the field could be read, false otherwise</returns>
public bool TryGetUnsignedIntegerField(string FieldName, out uint Result)
{
object RawValue;
if (!RawObject.TryGetValue(FieldName, out RawValue) || !uint.TryParse(RawValue.ToString(), out Result))
{
Result = 0;
return false;
}
return true;
}
/// <summary>
/// Gets a double field by the given name from the object, throwing an exception if it is not there or cannot be parsed.
/// </summary>
/// <param name="FieldName">Name of the field to get</param>
/// <returns>The field value</returns>
public double GetDoubleField(string FieldName)
{
double DoubleValue;
if (!TryGetDoubleField(FieldName, out DoubleValue))
{
throw new JsonParseException("Missing or invalid '{0}' field", FieldName);
}
return DoubleValue;
}
/// <summary>
/// Tries to read a double field by the given name from the object
/// </summary>
/// <param name="FieldName">Name of the field to get</param>
/// <param name="Result">On success, receives the field value</param>
/// <returns>True if the field could be read, false otherwise</returns>
public bool TryGetDoubleField(string FieldName, out double Result)
{
object RawValue;
if (!RawObject.TryGetValue(FieldName, out RawValue) || !double.TryParse(RawValue.ToString(), out Result))
{
Result = 0.0;
return false;
}
return true;
}
/// <summary>
/// Gets an enum field by the given name from the object, throwing an exception if it is not there or cannot be parsed.
/// </summary>
/// <param name="FieldName">Name of the field to get</param>
/// <returns>The field value</returns>
public T GetEnumField<T>(string FieldName) where T : struct
{
T EnumValue;
if (!TryGetEnumField(FieldName, out EnumValue))
{
throw new JsonParseException("Missing or invalid '{0}' field", FieldName);
}
return EnumValue;
}
/// <summary>
/// Tries to read an enum field by the given name from the object
/// </summary>
/// <param name="FieldName">Name of the field to get</param>
/// <param name="Result">On success, receives the field value</param>
/// <returns>True if the field could be read, false otherwise</returns>
public bool TryGetEnumField<T>(string FieldName, out T Result) where T : struct
{
string StringValue;
if (!TryGetStringField(FieldName, out StringValue) || !Enum.TryParse<T>(StringValue, true, out Result))
{
Result = default(T);
return false;
}
return true;
}
/// <summary>
/// Tries to read an enum array field by the given name from the object
/// </summary>
/// <param name="FieldName">Name of the field to get</param>
/// <param name="Result">On success, receives the field value</param>
/// <returns>True if the field could be read, false otherwise</returns>
public bool TryGetEnumArrayField<T>(string FieldName, out T[] Result) where T : struct
{
string[] StringValues;
if (!TryGetStringArrayField(FieldName, out StringValues))
{
Result = null;
return false;
}
T[] EnumValues = new T[StringValues.Length];
for (int Idx = 0; Idx < StringValues.Length; Idx++)
{
if (!Enum.TryParse<T>(StringValues[Idx], true, out EnumValues[Idx]))
{
Result = null;
return false;
}
}
Result = EnumValues;
return true;
}
/// <summary>
/// Gets an object field by the given name from the object, throwing an exception if it is not there or cannot be parsed.
/// </summary>
/// <param name="FieldName">Name of the field to get</param>
/// <returns>The field value</returns>
public JsonObject GetObjectField(string FieldName)
{
JsonObject Result;
if (!TryGetObjectField(FieldName, out Result))
{
throw new JsonParseException("Missing or invalid '{0}' field", FieldName);
}
return Result;
}
/// <summary>
/// Tries to read an object field by the given name from the object
/// </summary>
/// <param name="FieldName">Name of the field to get</param>
/// <param name="Result">On success, receives the field value</param>
/// <returns>True if the field could be read, false otherwise</returns>
public bool TryGetObjectField(string FieldName, out JsonObject Result)
{
object RawValue;
if (RawObject.TryGetValue(FieldName, out RawValue) && (RawValue is Dictionary<string, object>))
{
Result = new JsonObject((Dictionary<string, object>)RawValue);
return true;
}
else
{
Result = null;
return false;
}
}
/// <summary>
/// Gets an object array field by the given name from the object, throwing an exception if it is not there or cannot be parsed.
/// </summary>
/// <param name="FieldName">Name of the field to get</param>
/// <returns>The field value</returns>
public JsonObject[] GetObjectArrayField(string FieldName)
{
JsonObject[] Result;
if (!TryGetObjectArrayField(FieldName, out Result))
{
throw new JsonParseException("Missing or invalid '{0}' field", FieldName);
}
return Result;
}
/// <summary>
/// Tries to read an object array field by the given name from the object
/// </summary>
/// <param name="FieldName">Name of the field to get</param>
/// <param name="Result">On success, receives the field value</param>
/// <returns>True if the field could be read, false otherwise</returns>
public bool TryGetObjectArrayField(string FieldName, out JsonObject[] Result)
{
object RawValue;
if (RawObject.TryGetValue(FieldName, out RawValue) && (RawValue is IEnumerable<object>) && ((IEnumerable<object>)RawValue).All(x => x is Dictionary<string, object>))
{
Result = ((IEnumerable<object>)RawValue).Select(x => new JsonObject((Dictionary<string, object>)x)).ToArray();
return true;
}
else
{
Result = null;
return false;
}
}
}
/// <summary>
/// Writer for JSON data, which indents the output text appropriately, and adds commas and newlines between fields
/// </summary>
public class JsonWriter : IDisposable
{
TextWriter Writer;
bool bLeaveOpen;
bool bRequiresComma;
string Indent;
/// <summary>
/// Constructor
/// </summary>
/// <param name="FileName">File to write to</param>
public JsonWriter(string FileName)
: this(new StreamWriter(FileName))
{
}
/// <summary>
/// Constructor
/// </summary>
/// <param name="FileName">File to write to</param>
public JsonWriter(FileReference FileName)
: this(new StreamWriter(FileName.FullName))
{
}
/// <summary>
/// Constructor
/// </summary>
/// <param name="Writer">The text writer to output to</param>
/// <param name="bLeaveOpen">Whether to leave the writer open when the object is disposed</param>
public JsonWriter(TextWriter Writer, bool bLeaveOpen = false)
{
this.Writer = Writer;
this.bLeaveOpen = bLeaveOpen;
Indent = "";
}
/// <summary>
/// Dispose of any managed resources
/// </summary>
public void Dispose()
{
if(!bLeaveOpen && Writer != null)
{
Writer.Dispose();
Writer = null;
}
}
/// <summary>
/// Write the opening brace for an object
/// </summary>
public void WriteObjectStart()
{
WriteCommaNewline();
Writer.Write(Indent);
Writer.Write("{");
Indent += "\t";
bRequiresComma = false;
}
/// <summary>
/// Write the name and opening brace for an object
/// </summary>
/// <param name="ObjectName">Name of the field</param>
public void WriteObjectStart(string ObjectName)
{
WriteCommaNewline();
Writer.Write("{0}\"{1}\": ", Indent, ObjectName);
bRequiresComma = false;
WriteObjectStart();
}
/// <summary>
/// Write the closing brace for an object
/// </summary>
public void WriteObjectEnd()
{
Indent = Indent.Substring(0, Indent.Length - 1);
Writer.WriteLine();
Writer.Write(Indent);
Writer.Write("}");
bRequiresComma = true;
}
/// <summary>
/// Write the name and opening bracket for an array
/// </summary>
/// <param name="ArrayName">Name of the field</param>
public void WriteArrayStart(string ArrayName)
{
WriteCommaNewline();
Writer.Write("{0}\"{1}\": [", Indent, ArrayName);
Indent += "\t";
bRequiresComma = false;
}
/// <summary>
/// Write the closing bracket for an array
/// </summary>
public void WriteArrayEnd()
{
Indent = Indent.Substring(0, Indent.Length - 1);
Writer.WriteLine();
Writer.Write("{0}]", Indent);
bRequiresComma = true;
}
/// <summary>
/// Write an array of strings
/// </summary>
/// <param name="Name">Name of the field</param>
/// <param name="Values">Values for the field</param>
public void WriteStringArrayField(string Name, IEnumerable<string> Values)
{
WriteArrayStart(Name);
foreach(string Value in Values)
{
WriteValue(Value);
}
WriteArrayEnd();
}
/// <summary>
/// Write an array of enum values
/// </summary>
/// <param name="Name">Name of the field</param>
/// <param name="Values">Values for the field</param>
public void WriteEnumArrayField<T>(string Name, IEnumerable<T> Values) where T : struct
{
WriteStringArrayField(Name, Values.Select(x => x.ToString()));
}
/// <summary>
/// Write a value with no field name, for the contents of an array
/// </summary>
/// <param name="Value">Value to write</param>
public void WriteValue(string Value)
{
WriteCommaNewline();
Writer.Write(Indent);
WriteEscapedString(Value);
bRequiresComma = true;
}
/// <summary>
/// Write a field name and string value
/// </summary>
/// <param name="Name">Name of the field</param>
/// <param name="Value">Value for the field</param>
public void WriteValue(string Name, string Value)
{
WriteCommaNewline();
Writer.Write("{0}\"{1}\": ", Indent, Name);
WriteEscapedString(Value);
bRequiresComma = true;
}
/// <summary>
/// Write a field name and integer value
/// </summary>
/// <param name="Name">Name of the field</param>
/// <param name="Value">Value for the field</param>
public void WriteValue(string Name, int Value)
{
WriteValueInternal(Name, Value.ToString());
}
/// <summary>
/// Write a field name and double value
/// </summary>
/// <param name="Name">Name of the field</param>
/// <param name="Value">Value for the field</param>
public void WriteValue(string Name, double Value)
{
WriteValueInternal(Name, Value.ToString());
}
/// <summary>
/// Write a field name and bool value
/// </summary>
/// <param name="Name">Name of the field</param>
/// <param name="Value">Value for the field</param>
public void WriteValue(string Name, bool Value)
{
WriteValueInternal(Name, Value ? "true" : "false");
}
void WriteCommaNewline()
{
if (bRequiresComma)
{
Writer.WriteLine(",");
}
else if (Indent.Length > 0)
{
Writer.WriteLine();
}
}
void WriteValueInternal(string Name, string Value)
{
WriteCommaNewline();
Writer.Write("{0}\"{1}\": {2}", Indent, Name, Value);
bRequiresComma = true;
}
void WriteEscapedString(string Value)
{
// Escape any characters which may not appear in a JSON string (see http://www.json.org).
Writer.Write("\"");
if (Value != null)
{
for (int Idx = 0; Idx < Value.Length; Idx++)
{
switch (Value[Idx])
{
case '\"':
Writer.Write("\\\"");
break;
case '\\':
Writer.Write("\\\\");
break;
case '\b':
Writer.Write("\\b");
break;
case '\f':
Writer.Write("\\f");
break;
case '\n':
Writer.Write("\\n");
break;
case '\r':
Writer.Write("\\r");
break;
case '\t':
Writer.Write("\\t");
break;
default:
if (Char.IsControl(Value[Idx]))
{
Writer.Write("\\u{0:X4}", (int)Value[Idx]);
}
else
{
Writer.Write(Value[Idx]);
}
break;
}
}
}
Writer.Write("\"");
}
}
}