-
-
Notifications
You must be signed in to change notification settings - Fork 706
/
MessagePackSerializerOptions.cs
314 lines (278 loc) · 13.7 KB
/
MessagePackSerializerOptions.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
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
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
// Copyright (c) All contributors. All rights reserved.
// Licensed under the MIT license. See LICENSE file in the project root for full license information.
using System;
using System.Collections.Generic;
using System.Text.RegularExpressions;
using Nerdbank.Streams;
namespace MessagePack
{
#if !DYNAMICCODEDUMPER
/// <summary>
/// An immutable description of options for running the <see cref="MessagePackSerializer"/>.
/// </summary>
#endif
public class MessagePackSerializerOptions
{
// see:http://msdn.microsoft.com/en-us/library/w3f99sx1.aspx
internal static readonly Regex AssemblyNameVersionSelectorRegex = new Regex(@", Version=\d+.\d+.\d+.\d+, Culture=[\w-]+, PublicKeyToken=(?:null|[a-f0-9]{16})$", RegexOptions.Compiled);
/// <summary>
/// A collection of known dangerous types that are not expected in a typical MessagePack stream,
/// and thus are rejected by the default implementation of <see cref="ThrowIfDeserializingTypeIsDisallowed(Type)"/>.
/// </summary>
private static readonly HashSet<string> DisallowedTypes = new HashSet<string>
{
"System.CodeDom.Compiler.TempFileCollection",
"System.Management.IWbemClassObjectFreeThreaded",
};
#if !DYNAMICCODEDUMPER
/// <summary>
/// Gets a good default set of options that uses the <see cref="Resolvers.StandardResolver"/> and no compression.
/// </summary>
public static MessagePackSerializerOptions Standard => MessagePackSerializerOptionsDefaultSettingsLazyInitializationHelper.Standard;
#endif
/// <summary>
/// Initializes a new instance of the <see cref="MessagePackSerializerOptions"/> class.
/// </summary>
protected internal MessagePackSerializerOptions(IFormatterResolver resolver)
{
this.Resolver = resolver ?? throw new ArgumentNullException(nameof(resolver));
}
/// <summary>
/// Initializes a new instance of the <see cref="MessagePackSerializerOptions"/> class
/// with members initialized from an existing instance.
/// </summary>
/// <param name="copyFrom">The options to copy from.</param>
protected MessagePackSerializerOptions(MessagePackSerializerOptions copyFrom)
{
if (copyFrom == null)
{
throw new ArgumentNullException(nameof(copyFrom));
}
this.Resolver = copyFrom.Resolver;
this.Compression = copyFrom.Compression;
this.OldSpec = copyFrom.OldSpec;
this.OmitAssemblyVersion = copyFrom.OmitAssemblyVersion;
this.AllowAssemblyVersionMismatch = copyFrom.AllowAssemblyVersionMismatch;
this.Security = copyFrom.Security;
this.SequencePool = copyFrom.SequencePool;
}
/// <summary>
/// Gets the resolver to use for complex types.
/// </summary>
/// <value>An instance of <see cref="IFormatterResolver"/>. Never <c>null</c>.</value>
/// <exception cref="ArgumentNullException">Thrown if an attempt is made to set this property to <c>null</c>.</exception>
public IFormatterResolver Resolver { get; private set; }
/// <summary>
/// Gets the compression scheme to apply to serialized sequences.
/// </summary>
/// <remarks>
/// When set to something other than <see cref="MessagePackCompression.None"/>,
/// deserialization can still work on uncompressed sequences,
/// and serialization may not compress if msgpack sequences are short enough that compression would not likely be advantageous.
/// </remarks>
public MessagePackCompression Compression { get; private set; }
/// <summary>
/// Gets a value indicating whether to serialize with <see cref="MessagePackWriter.OldSpec"/> set to some value
/// causing messagepack spec compliance to be explicitly set to the old or new format.
/// </summary>
/// <value>
/// A null value means the <see cref="MessagePackWriter"/>'s default or previous setting will be used.
/// A non-null value will be applied to the <see cref="MessagePackWriter.OldSpec"/> property for the duration of a
/// serialization and then reverted to its prior setting.
/// </value>
/// <remarks>
/// Reading always supports both new and old spec.
/// </remarks>
public bool? OldSpec { get; private set; }
/// <summary>
/// Gets a value indicating whether serialization should omit assembly version, culture and public key token metadata when using the typeless formatter.
/// </summary>
/// <value>The default value is <c>false</c>.</value>
public bool OmitAssemblyVersion { get; private set; }
/// <summary>
/// Gets a value indicating whether deserialization may instantiate types from an assembly with a different version if a matching version cannot be found.
/// </summary>
/// <value>The default value is <c>false</c>.</value>
public bool AllowAssemblyVersionMismatch { get; private set; }
/// <summary>
/// Gets the security-related options for deserializing messagepack sequences.
/// </summary>
/// <value>
/// The default value is to use <see cref="MessagePackSecurity.TrustedData"/>.
/// </value>
public MessagePackSecurity Security { get; private set; } = MessagePackSecurity.TrustedData;
/// <summary>
/// Gets a thread-safe pool of reusable <see cref="Sequence{T}"/> objects.
/// </summary>
/// <value>The default value is the <see cref="SequencePool.Shared"/> instance.</value>
public SequencePool SequencePool { get; private set; } = SequencePool.Shared;
/// <summary>
/// Gets a type given a string representation of the type.
/// </summary>
/// <param name="typeName">The name of the type to load. This is typically the <see cref="Type.AssemblyQualifiedName"/> but may use the assembly's simple name.</param>
/// <returns>The loaded type or <c>null</c> if no matching type could be found.</returns>
public virtual Type LoadType(string typeName)
{
Type result = Type.GetType(typeName, false);
if (result == null && this.AllowAssemblyVersionMismatch)
{
string shortenedName = AssemblyNameVersionSelectorRegex.Replace(typeName, string.Empty);
if (shortenedName != typeName)
{
result = Type.GetType(shortenedName, false);
}
}
return result;
}
/// <summary>
/// Checks whether a given type may be deserialized.
/// </summary>
/// <param name="type">The type to be instantiated.</param>
/// <exception cref="TypeAccessException">Thrown if the <paramref name="type"/> is not allowed to be deserialized.</exception>
/// <remarks>
/// This method provides a means for an important security mitigation when using the Typeless formatter to prevent untrusted messagepack from
/// deserializing objects that may be harmful if instantiated, disposed or finalized.
/// The default implementation throws for only a few known dangerous types.
/// Applications that deserialize from untrusted sources should override this method and throw if the type is not among the expected set.
/// </remarks>
public virtual void ThrowIfDeserializingTypeIsDisallowed(Type type)
{
if (DisallowedTypes.Contains(type.FullName))
{
throw new MessagePackSerializationException("Deserialization attempted to create the type " + type.FullName + " which is not allowed.");
}
}
/// <summary>
/// Gets a copy of these options with the <see cref="Resolver"/> property set to a new value.
/// </summary>
/// <param name="resolver">The new value for the <see cref="Resolver"/>.</param>
/// <returns>The new instance; or the original if the value is unchanged.</returns>
public MessagePackSerializerOptions WithResolver(IFormatterResolver resolver)
{
if (this.Resolver == resolver)
{
return this;
}
var result = this.Clone();
result.Resolver = resolver;
return result;
}
/// <summary>
/// Gets a copy of these options with the <see cref="Compression"/> property set to a new value.
/// </summary>
/// <param name="compression">The new value for the <see cref="Compression"/> property.</param>
/// <returns>The new instance; or the original if the value is unchanged.</returns>
public MessagePackSerializerOptions WithCompression(MessagePackCompression compression)
{
if (this.Compression == compression)
{
return this;
}
var result = this.Clone();
result.Compression = compression;
return result;
}
/// <summary>
/// Gets a copy of these options with the <see cref="OldSpec"/> property set to a new value.
/// </summary>
/// <param name="oldSpec">The new value for the <see cref="OldSpec"/>.</param>
/// <returns>The new instance; or the original if the value is unchanged.</returns>
public MessagePackSerializerOptions WithOldSpec(bool? oldSpec = true)
{
if (this.OldSpec == oldSpec)
{
return this;
}
var result = this.Clone();
result.OldSpec = oldSpec;
return result;
}
/// <summary>
/// Gets a copy of these options with the <see cref="OmitAssemblyVersion"/> property set to a new value.
/// </summary>
/// <param name="omitAssemblyVersion">The new value for the <see cref="OmitAssemblyVersion"/> property.</param>
/// <returns>The new instance; or the original if the value is unchanged.</returns>
public MessagePackSerializerOptions WithOmitAssemblyVersion(bool omitAssemblyVersion)
{
if (this.OmitAssemblyVersion == omitAssemblyVersion)
{
return this;
}
var result = this.Clone();
result.OmitAssemblyVersion = omitAssemblyVersion;
return result;
}
/// <summary>
/// Gets a copy of these options with the <see cref="AllowAssemblyVersionMismatch"/> property set to a new value.
/// </summary>
/// <param name="allowAssemblyVersionMismatch">The new value for the <see cref="AllowAssemblyVersionMismatch"/> property.</param>
/// <returns>The new instance; or the original if the value is unchanged.</returns>
public MessagePackSerializerOptions WithAllowAssemblyVersionMismatch(bool allowAssemblyVersionMismatch)
{
if (this.AllowAssemblyVersionMismatch == allowAssemblyVersionMismatch)
{
return this;
}
var result = this.Clone();
result.AllowAssemblyVersionMismatch = allowAssemblyVersionMismatch;
return result;
}
/// <summary>
/// Gets a copy of these options with the <see cref="Security"/> property set to a new value.
/// </summary>
/// <param name="security">The new value for the <see cref="Security"/> property.</param>
/// <returns>The new instance; or the original if the value is unchanged.</returns>
public MessagePackSerializerOptions WithSecurity(MessagePackSecurity security)
{
if (security is null)
{
throw new ArgumentNullException(nameof(security));
}
if (this.Security == security)
{
return this;
}
var result = this.Clone();
result.Security = security;
return result;
}
/// <summary>
/// Gets a copy of these options with the <see cref="SequencePool"/> property set to a new value.
/// </summary>
/// <param name="pool">The new value for the <see cref="SequencePool"/> property.</param>
/// <returns>The new instance.</returns>
public MessagePackSerializerOptions WithPool(SequencePool pool)
{
if (pool is null)
{
throw new ArgumentNullException(nameof(pool));
}
if (this.SequencePool == pool)
{
return this;
}
var result = this.Clone();
result.SequencePool = pool;
return result;
}
/// <summary>
/// Creates a clone of this instance with the same properties set.
/// </summary>
/// <returns>The cloned instance. Guaranteed to be a new instance.</returns>
/// <exception cref="NotSupportedException">Thrown if this instance is a derived type that doesn't override this method.</exception>
protected virtual MessagePackSerializerOptions Clone()
{
if (this.GetType() != typeof(MessagePackSerializerOptions))
{
throw new NotSupportedException($"The derived type {this.GetType().FullName} did not override the {nameof(Clone)} method as required.");
}
return new MessagePackSerializerOptions(this);
}
#if !DYNAMICCODEDUMPER
private static class MessagePackSerializerOptionsDefaultSettingsLazyInitializationHelper
{
public static readonly MessagePackSerializerOptions Standard = new MessagePackSerializerOptions(Resolvers.StandardResolver.Instance);
}
#endif
}
}