JsonSchemaBuilder
class JsonSchemaBuilder
using Newtonsoft.Json.Linq;
using Newtonsoft.Json.Utilities;
using Newtonsoft.Json.Utilities.LinqBridge;
using System;
using System.Collections.Generic;
using System.Globalization;
namespace Newtonsoft.Json.Schema
{
[Obsolete("JSON Schema validation has been moved to its own package. See https://www.newtonsoft.com/jsonschema for more details.")]
internal class JsonSchemaBuilder
{
private readonly IList<JsonSchema> _stack;
private readonly JsonSchemaResolver _resolver;
private readonly IDictionary<string, JsonSchema> _documentSchemas;
private JsonSchema _currentSchema;
private JObject _rootSchema;
private JsonSchema CurrentSchema => _currentSchema;
public JsonSchemaBuilder(JsonSchemaResolver resolver)
{
_stack = new List<JsonSchema>();
_documentSchemas = new Dictionary<string, JsonSchema>();
_resolver = resolver;
}
private void Push(JsonSchema value)
{
_currentSchema = value;
_stack.Add(value);
_resolver.LoadedSchemas.Add(value);
_documentSchemas.Add(value.Location, value);
}
private JsonSchema Pop()
{
JsonSchema currentSchema = _currentSchema;
_stack.RemoveAt(_stack.Count - 1);
_currentSchema = _stack.LastOrDefault();
return currentSchema;
}
internal JsonSchema Read(JsonReader reader)
{
JToken jToken = JToken.ReadFrom(reader);
_rootSchema = (jToken as JObject);
JsonSchema jsonSchema = BuildSchema(jToken);
ResolveReferences(jsonSchema);
return jsonSchema;
}
private string UnescapeReference(string reference)
{
return StringUtils.Replace(StringUtils.Replace(Uri.UnescapeDataString(reference), "~1", "/"), "~0", "~");
}
private JsonSchema ResolveReferences(JsonSchema schema)
{
if (schema.DeferredReference != null) {
string text = schema.DeferredReference;
bool flag = text.StartsWith("#", StringComparison.Ordinal);
if (flag)
text = UnescapeReference(text);
JsonSchema jsonSchema = _resolver.GetSchema(text);
if (jsonSchema == null) {
if (flag) {
string[] array = schema.DeferredReference.TrimStart(new char[1] {
'#'
}).Split(new char[1] {
'/'
}, StringSplitOptions.RemoveEmptyEntries);
JToken jToken = _rootSchema;
string[] array2 = array;
foreach (string reference in array2) {
string text2 = UnescapeReference(reference);
if (jToken.Type == JTokenType.Object)
jToken = jToken[text2];
else if (jToken.Type == JTokenType.Array || jToken.Type == JTokenType.Constructor) {
int result;
jToken = ((!int.TryParse(text2, out result) || result < 0 || result >= jToken.Count()) ? null : jToken[result]);
}
if (jToken == null)
break;
}
if (jToken != null)
jsonSchema = BuildSchema(jToken);
}
if (jsonSchema == null)
throw new JsonException("Could not resolve schema reference '{0}'.".FormatWith(CultureInfo.InvariantCulture, schema.DeferredReference));
}
schema = jsonSchema;
}
if (schema.ReferencesResolved)
return schema;
schema.ReferencesResolved = true;
if (schema.Extends != null) {
for (int j = 0; j < schema.Extends.Count; j++) {
schema.Extends[j] = ResolveReferences(schema.Extends[j]);
}
}
if (schema.Items != null) {
for (int k = 0; k < schema.Items.Count; k++) {
schema.Items[k] = ResolveReferences(schema.Items[k]);
}
}
if (schema.AdditionalItems != null)
schema.AdditionalItems = ResolveReferences(schema.AdditionalItems);
List<KeyValuePair<string, JsonSchema>>.Enumerator enumerator;
if (schema.PatternProperties != null) {
enumerator = schema.PatternProperties.ToList().GetEnumerator();
try {
while (enumerator.MoveNext()) {
KeyValuePair<string, JsonSchema> current = enumerator.Current;
schema.PatternProperties[current.Key] = ResolveReferences(current.Value);
}
} finally {
((IDisposable)enumerator).Dispose();
}
}
if (schema.Properties != null) {
enumerator = schema.Properties.ToList().GetEnumerator();
try {
while (enumerator.MoveNext()) {
KeyValuePair<string, JsonSchema> current2 = enumerator.Current;
schema.Properties[current2.Key] = ResolveReferences(current2.Value);
}
} finally {
((IDisposable)enumerator).Dispose();
}
}
if (schema.AdditionalProperties != null)
schema.AdditionalProperties = ResolveReferences(schema.AdditionalProperties);
return schema;
}
private JsonSchema BuildSchema(JToken token)
{
JObject jObject = token as JObject;
if (jObject == null)
throw JsonException.Create(token, token.Path, "Expected object while parsing schema object, got {0}.".FormatWith(CultureInfo.InvariantCulture, token.Type));
if (jObject.TryGetValue("$ref", out JToken value))
return new JsonSchema {
DeferredReference = (string)value
};
string path = token.Path;
path = StringUtils.Replace(path, ".", "/");
path = StringUtils.Replace(path, "[", "/");
path = StringUtils.Replace(path, "]", string.Empty);
if (!StringUtils.IsNullOrEmpty(path))
path = "/" + path;
path = "#" + path;
if (_documentSchemas.TryGetValue(path, out JsonSchema value2))
return value2;
Push(new JsonSchema {
Location = path
});
ProcessSchemaProperties(jObject);
return Pop();
}
private void ProcessSchemaProperties(JObject schemaObject)
{
foreach (KeyValuePair<string, JToken> item in schemaObject) {
switch (item.Key) {
case "type":
CurrentSchema.Type = ProcessType(item.Value);
break;
case "id":
CurrentSchema.Id = (string)item.Value;
break;
case "title":
CurrentSchema.Title = (string)item.Value;
break;
case "description":
CurrentSchema.Description = (string)item.Value;
break;
case "properties":
CurrentSchema.Properties = ProcessProperties(item.Value);
break;
case "items":
ProcessItems(item.Value);
break;
case "additionalProperties":
ProcessAdditionalProperties(item.Value);
break;
case "additionalItems":
ProcessAdditionalItems(item.Value);
break;
case "patternProperties":
CurrentSchema.PatternProperties = ProcessProperties(item.Value);
break;
case "required":
CurrentSchema.Required = (bool)item.Value;
break;
case "requires":
CurrentSchema.Requires = (string)item.Value;
break;
case "minimum":
CurrentSchema.Minimum = (double)item.Value;
break;
case "maximum":
CurrentSchema.Maximum = (double)item.Value;
break;
case "exclusiveMinimum":
CurrentSchema.ExclusiveMinimum = (bool)item.Value;
break;
case "exclusiveMaximum":
CurrentSchema.ExclusiveMaximum = (bool)item.Value;
break;
case "maxLength":
CurrentSchema.MaximumLength = (int)item.Value;
break;
case "minLength":
CurrentSchema.MinimumLength = (int)item.Value;
break;
case "maxItems":
CurrentSchema.MaximumItems = (int)item.Value;
break;
case "minItems":
CurrentSchema.MinimumItems = (int)item.Value;
break;
case "divisibleBy":
CurrentSchema.DivisibleBy = (double)item.Value;
break;
case "disallow":
CurrentSchema.Disallow = ProcessType(item.Value);
break;
case "default":
CurrentSchema.Default = item.Value.DeepClone();
break;
case "hidden":
CurrentSchema.Hidden = (bool)item.Value;
break;
case "readonly":
CurrentSchema.ReadOnly = (bool)item.Value;
break;
case "format":
CurrentSchema.Format = (string)item.Value;
break;
case "pattern":
CurrentSchema.Pattern = (string)item.Value;
break;
case "enum":
ProcessEnum(item.Value);
break;
case "extends":
ProcessExtends(item.Value);
break;
case "uniqueItems":
CurrentSchema.UniqueItems = (bool)item.Value;
break;
}
}
}
private void ProcessExtends(JToken token)
{
IList<JsonSchema> list = new List<JsonSchema>();
if (token.Type == JTokenType.Array) {
foreach (JToken item in (IEnumerable<JToken>)token) {
list.Add(BuildSchema(item));
}
} else {
JsonSchema jsonSchema = BuildSchema(token);
if (jsonSchema != null)
list.Add(jsonSchema);
}
if (list.Count > 0)
CurrentSchema.Extends = list;
}
private void ProcessEnum(JToken token)
{
if (token.Type != JTokenType.Array)
throw JsonException.Create(token, token.Path, "Expected Array token while parsing enum values, got {0}.".FormatWith(CultureInfo.InvariantCulture, token.Type));
CurrentSchema.Enum = new List<JToken>();
foreach (JToken item in (IEnumerable<JToken>)token) {
CurrentSchema.Enum.Add(item.DeepClone());
}
}
private void ProcessAdditionalProperties(JToken token)
{
if (token.Type == JTokenType.Boolean)
CurrentSchema.AllowAdditionalProperties = (bool)token;
else
CurrentSchema.AdditionalProperties = BuildSchema(token);
}
private void ProcessAdditionalItems(JToken token)
{
if (token.Type == JTokenType.Boolean)
CurrentSchema.AllowAdditionalItems = (bool)token;
else
CurrentSchema.AdditionalItems = BuildSchema(token);
}
private IDictionary<string, JsonSchema> ProcessProperties(JToken token)
{
IDictionary<string, JsonSchema> dictionary = new Dictionary<string, JsonSchema>();
if (token.Type != JTokenType.Object)
throw JsonException.Create(token, token.Path, "Expected Object token while parsing schema properties, got {0}.".FormatWith(CultureInfo.InvariantCulture, token.Type));
foreach (JProperty item in (IEnumerable<JToken>)token) {
if (dictionary.ContainsKey(item.Name))
throw new JsonException("Property {0} has already been defined in schema.".FormatWith(CultureInfo.InvariantCulture, item.Name));
dictionary.Add(item.Name, BuildSchema(item.Value));
}
return dictionary;
}
private void ProcessItems(JToken token)
{
CurrentSchema.Items = new List<JsonSchema>();
switch (token.Type) {
case JTokenType.Object:
CurrentSchema.Items.Add(BuildSchema(token));
CurrentSchema.PositionalItemsValidation = false;
break;
case JTokenType.Array:
CurrentSchema.PositionalItemsValidation = true;
foreach (JToken item in (IEnumerable<JToken>)token) {
CurrentSchema.Items.Add(BuildSchema(item));
}
break;
default:
throw JsonException.Create(token, token.Path, "Expected array or JSON schema object, got {0}.".FormatWith(CultureInfo.InvariantCulture, token.Type));
}
}
private JsonSchemaType? ProcessType(JToken token)
{
switch (token.Type) {
case JTokenType.Array: {
JsonSchemaType? nullable = JsonSchemaType.None;
{
foreach (JToken item in (IEnumerable<JToken>)token) {
if (item.Type != JTokenType.String)
throw JsonException.Create(item, item.Path, "Expected JSON schema type string token, got {0}.".FormatWith(CultureInfo.InvariantCulture, token.Type));
nullable |= MapType((string)item);
}
return nullable;
}
}
case JTokenType.String:
return MapType((string)token);
default:
throw JsonException.Create(token, token.Path, "Expected array or JSON schema type string token, got {0}.".FormatWith(CultureInfo.InvariantCulture, token.Type));
}
}
internal static JsonSchemaType MapType(string type)
{
if (!JsonSchemaConstants.JsonSchemaTypeMapping.TryGetValue(type, out JsonSchemaType value))
throw new JsonException("Invalid JSON schema type: {0}".FormatWith(CultureInfo.InvariantCulture, type));
return value;
}
internal static string MapType(JsonSchemaType type)
{
return JsonSchemaConstants.JsonSchemaTypeMapping.Single((KeyValuePair<string, JsonSchemaType> kv) => kv.Value == type).Key;
}
}
}