-
Notifications
You must be signed in to change notification settings - Fork 4.7k
/
JsonTestHelper.cs
191 lines (169 loc) · 7.57 KB
/
JsonTestHelper.cs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.
using System.Collections.Generic;
using System.Diagnostics;
using System.Linq;
using System.Text.RegularExpressions;
using System.Threading.Tasks;
using Xunit;
namespace System.Text.Json
{
internal static partial class JsonTestHelper
{
public static void AssertJsonEqual(string expected, string actual)
{
using JsonDocument expectedDom = JsonDocument.Parse(expected);
using JsonDocument actualDom = JsonDocument.Parse(actual);
AssertJsonEqual(expectedDom.RootElement, actualDom.RootElement);
}
public static void AssertJsonEqual(JsonElement expected, JsonElement actual)
{
AssertJsonEqualCore(expected, actual, new());
}
private static void AssertJsonEqualCore(JsonElement expected, JsonElement actual, Stack<object> path)
{
JsonValueKind valueKind = expected.ValueKind;
AssertTrue(passCondition: valueKind == actual.ValueKind);
switch (valueKind)
{
case JsonValueKind.Object:
var expectedProperties = new List<string>();
foreach (JsonProperty property in expected.EnumerateObject())
{
expectedProperties.Add(property.Name);
}
var actualProperties = new List<string>();
foreach (JsonProperty property in actual.EnumerateObject())
{
actualProperties.Add(property.Name);
}
foreach (var property in expectedProperties.Except(actualProperties))
{
AssertTrue(passCondition: false, $"Property \"{property}\" missing from actual object.");
}
foreach (var property in actualProperties.Except(expectedProperties))
{
AssertTrue(passCondition: false, $"Actual object defines additional property \"{property}\".");
}
foreach (string name in expectedProperties)
{
path.Push(name);
AssertJsonEqualCore(expected.GetProperty(name), actual.GetProperty(name), path);
path.Pop();
}
break;
case JsonValueKind.Array:
JsonElement.ArrayEnumerator expectedEnumerator = expected.EnumerateArray();
JsonElement.ArrayEnumerator actualEnumerator = actual.EnumerateArray();
int i = 0;
while (expectedEnumerator.MoveNext())
{
AssertTrue(passCondition: actualEnumerator.MoveNext(), "Actual array contains fewer elements.");
path.Push(i++);
AssertJsonEqualCore(expectedEnumerator.Current, actualEnumerator.Current, path);
path.Pop();
}
AssertTrue(passCondition: !actualEnumerator.MoveNext(), "Actual array contains additional elements.");
break;
case JsonValueKind.String:
AssertTrue(passCondition: expected.GetString() == actual.GetString());
break;
case JsonValueKind.Number:
case JsonValueKind.True:
case JsonValueKind.False:
case JsonValueKind.Null:
AssertTrue(passCondition: expected.GetRawText() == actual.GetRawText());
break;
default:
Debug.Fail($"Unexpected JsonValueKind: JsonValueKind.{valueKind}.");
break;
}
void AssertTrue(bool passCondition, string? message = null)
{
if (!passCondition)
{
message ??= "Expected JSON does not match actual value";
Assert.Fail($"{message}\nExpected JSON: {expected}\n Actual JSON: {actual}\n in JsonPath: {BuildJsonPath(path)}");
}
// TODO replace with JsonPath implementation for JsonElement
// cf. https://github.com/dotnet/runtime/issues/31068
static string BuildJsonPath(Stack<object> path)
{
var sb = new StringBuilder("$");
foreach (object node in path.Reverse())
{
string pathNode = node is string propertyName
? "." + propertyName
: $"[{(int)node}]";
sb.Append(pathNode);
}
return sb.ToString();
}
}
}
/// <summary>
/// Linq Cartesian product
/// </summary>
public static IEnumerable<(TFirst First, TSecond Second)> CrossJoin<TFirst, TSecond>(this IEnumerable<TFirst> first, IEnumerable<TSecond> second)
{
TSecond[]? secondCached = null;
foreach (TFirst f in first)
{
secondCached ??= second.ToArray();
foreach (TSecond s in secondCached)
{
yield return (f, s);
}
}
}
/// <summary>
/// Linq Cartesian product
/// </summary>
public static IEnumerable<(TFirst First, TSecond Second, TThird Third)> CrossJoin<TFirst, TSecond, TThird>(
this IEnumerable<TFirst> first,
IEnumerable<TSecond> second,
IEnumerable<TThird> third)
{
TSecond[]? secondCached = null;
TThird[]? thirdCached = null;
foreach (TFirst f in first)
{
secondCached ??= second.ToArray();
foreach (TSecond s in secondCached)
{
thirdCached ??= third.ToArray();
foreach (TThird t in thirdCached)
{
yield return (f, s, t);
}
}
}
}
/// <summary>
/// Linq Cartesian product
/// </summary>
public static IEnumerable<TResult> CrossJoin<TFirst, TSecond, TResult>(this IEnumerable<TFirst> first, IEnumerable<TSecond> second, Func<TFirst, TSecond, TResult> resultSelector)
=> first.CrossJoin(second).Select(tuple => resultSelector(tuple.First, tuple.Second));
/// <summary>
/// Linq Cartesian product
/// </summary>
public static IEnumerable<TResult> CrossJoin<TFirst, TSecond, TThird, TResult>(
this IEnumerable<TFirst> first,
IEnumerable<TSecond> second,
IEnumerable<TThird> third,
Func<TFirst, TSecond, TThird, TResult> resultSelector)
=> first.CrossJoin(second, third).Select(tuple => resultSelector(tuple.First, tuple.Second, tuple.Third));
public static async Task<List<T>> ToListAsync<T>(this IAsyncEnumerable<T> source)
{
var list = new List<T>();
await foreach (T item in source)
{
list.Add(item);
}
return list;
}
private static readonly Regex s_stripWhitespace = new Regex(@"\s+", RegexOptions.Compiled);
public static string StripWhitespace(this string value)
=> s_stripWhitespace.Replace(value, string.Empty);
}
}