This repository was archived by the owner on Jan 12, 2024. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 55
Expand file tree
/
Copy pathTupleConversion.cs
More file actions
220 lines (186 loc) · 7.98 KB
/
TupleConversion.cs
File metadata and controls
220 lines (186 loc) · 7.98 KB
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
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
// Copyright (c) Microsoft Corporation.
// Licensed under the MIT License.
using Newtonsoft.Json;
using Newtonsoft.Json.Linq;
using Microsoft.Quantum.Simulation.Core;
using System.Diagnostics;
using System.IO;
using System.Diagnostics.CodeAnalysis;
using Microsoft.Quantum.Simulation.Simulators;
using DataModel = Microsoft.Quantum.Simulation.OpenSystems.DataModel;
namespace Microsoft.Quantum.IQSharp;
public static class JsonConverters
{
private static readonly ImmutableList<JsonConverter> tupleConverters;
public static JsonConverter[] TupleConverters => tupleConverters.ToArray();
public static JsonConverter[] AllConverters => tupleConverters.ToArray();
static JsonConverters()
{
tupleConverters = new JsonConverter[]
{
new QTupleConverter(),
new QVoidConverter(),
new UDTConverter(),
new ResultConverter(),
// Make sure to use the base type for each open systems / CHP
// state, since these are effectively a discriminated union.
new DelegatedConverter<DataModel.PureState, DataModel.State>(new DataModel.StateConverter()),
new DelegatedConverter<DataModel.MixedState, DataModel.State>(new DataModel.StateConverter()),
new DelegatedConverter<DataModel.StabilizerState, DataModel.State>(new DataModel.StateConverter())
}
.ToImmutableList();
}
/// <summary>
/// A helper method to read a json object and return it as a dictionary.
/// Only the immediate elements of the object are used as keys. Their values
/// are returned as json objects themselves.
/// </summary>
/// <returns></returns>
public static Dictionary<string, string> JsonToDict(string json)
{
var result = new Dictionary<string, string>();
var args = JObject.Parse(json);
foreach (var a in args)
{
result.Add(a.Key, a.Value?.ToString(Formatting.None));
}
return result;
}
}
public class UDTConverter : JsonConverter
{
public override bool CanRead => true;
public override bool CanWrite => true;
public override bool CanConvert(Type objectType) =>
objectType.IsSubclassOfGenericType(typeof(UDTBase<>));
public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)
{
// Create an instance of the base Data type and populate it with the jObject:
var data = Activator.CreateInstance(objectType.GetProperty("Data").PropertyType);
serializer.Populate(reader, data);
return Activator.CreateInstance(objectType, data);
}
public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer)
{
var objectType = value.GetType();
var dataType = objectType.GetProperty("Data").PropertyType;
var tempWriter = new StringWriter();
serializer.Serialize(tempWriter, objectType.GetProperty("Data").GetValue(value), dataType);
var token = JToken.Parse(tempWriter.ToString());
token["@type"] = objectType.FullName;
token.WriteTo(writer);
}
}
public class QTupleConverter : JsonConverter
{
public override bool CanRead => false;
public override bool CanConvert(Type objectType)
{
if (objectType == typeof(ValueTuple)) return true;
// If we've survived, we either have a nongeneric type which isn't
// a value tuple, or we have a generic value tuple.
if (!objectType.IsGenericType) return false;
// Now we can compare the generic type to each possible pattern for
// value tuples.
var genericType = objectType.GetGenericTypeDefinition();
return genericType == typeof(ValueTuple<>)
|| genericType == typeof(ValueTuple<,>)
|| genericType == typeof(ValueTuple<,,>)
|| genericType == typeof(ValueTuple<,,,>)
|| genericType == typeof(ValueTuple<,,,,>)
|| genericType == typeof(ValueTuple<,,,,,>)
|| genericType == typeof(ValueTuple<,,,,,,>)
|| genericType == typeof(ValueTuple<,,,,,,,>);
}
public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)
{
throw new NotImplementedException();
}
public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer)
{
var tokenData = new Dictionary<string, object>
{
["@type"] = "@tuple"
};
var itemOffset = 0;
while (true)
{
var type = value.GetType();
var nItems = type.GenericTypeArguments.Length;
// For tuples of more than 7 items, the .NET type is ValueTuple<T1,T2,T3,T4,T5,T6,T7,TRest>
const int maxTupleLength = 7;
foreach (var idx in Enumerable.Range(0, System.Math.Min(nItems, maxTupleLength)))
{
var field = type.GetField($"Item{idx + 1}");
Debug.Assert(field != null, $"Failed trying to look at field Item{idx + 1} of a value tuple with {nItems} type arguments, {type.FullName}.");
tokenData[$"Item{idx + itemOffset + 1}"] = field.GetValue(value);
}
if (nItems <= maxTupleLength)
{
break;
}
value = type.GetField("Rest").GetValue(value);
itemOffset += maxTupleLength;
}
// See https://github.com/JamesNK/Newtonsoft.Json/issues/386#issuecomment-421161191
// for why this works to pass through.
var token = JToken.FromObject(tokenData, serializer);
token.WriteTo(writer);
}
}
public class QVoidConverter : JsonConverter<QVoid>
{
public override QVoid ReadJson(JsonReader reader, Type objectType, QVoid existingValue, bool hasExistingValue, JsonSerializer serializer)
{
throw new NotImplementedException();
}
public override void WriteJson(JsonWriter writer, QVoid value, JsonSerializer serializer)
{
var token = JToken.FromObject(new Dictionary<string, object>
{
["@type"] = "tuple"
});
token.WriteTo(writer);
}
}
public class ResultConverter : JsonConverter<Result>
{
public override Result ReadJson(JsonReader reader, Type objectType, Result existingValue, bool hasExistingValue, JsonSerializer serializer)
{
throw new NotImplementedException();
}
public override void WriteJson(JsonWriter writer, Result value, JsonSerializer serializer)
{
// See https://github.com/JamesNK/Newtonsoft.Json/issues/386#issuecomment-421161191
// for why this works to pass through.
var token = JToken.FromObject(value.GetValue(), serializer);
token.WriteTo(writer);
}
}
/// <summary>
/// Delegates JSON conversion from Newtonsoft.Json to a converter
/// for System.Text.Json.
/// </summary>
internal class DelegatedConverter<TTarget, TDelegated> : JsonConverter<TTarget>
where TTarget : class, TDelegated
{
private readonly System.Text.Json.Serialization.JsonConverter<TDelegated> converter;
private readonly System.Text.Json.JsonSerializerOptions options = new System.Text.Json.JsonSerializerOptions();
public DelegatedConverter(System.Text.Json.Serialization.JsonConverter<TDelegated> converter)
{
this.converter = converter;
this.options.Converters.Add(converter);
}
public override TTarget ReadJson(JsonReader reader, Type objectType, [AllowNull] TTarget existingValue, bool hasExistingValue, JsonSerializer serializer)
{
var asJson = JToken.ReadFrom(reader).ToString();
var delegated = System.Text.Json.JsonSerializer.Deserialize<TDelegated>(asJson, options);
return (TTarget)delegated;
}
public override void WriteJson(JsonWriter writer, [AllowNull] TTarget value, JsonSerializer serializer)
{
var asJson = System.Text.Json.JsonSerializer.Serialize<TDelegated>(value, options);
var token = JToken.Parse(asJson);
token.WriteTo(writer);
}
}