-
Notifications
You must be signed in to change notification settings - Fork 329
/
Copy pathDefaultProtobufSerializer.cs
167 lines (149 loc) · 5.35 KB
/
DefaultProtobufSerializer.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
namespace EasyCaching.Serialization.Protobuf
{
using System;
using System.IO;
using System.Text;
using EasyCaching.Core.Internal;
using EasyCaching.Core.Serialization;
using ProtoBuf;
/// <summary>
/// Default protobuf serializer.
/// </summary>
public class DefaultProtobufSerializer : IEasyCachingSerializer
{
/// <summary>
/// The name.
/// </summary>
private readonly string _name;
/// <summary>
/// Initializes a new instance of the
/// <see cref="T:EasyCaching.Serialization.Protobuf.DefaultProtobufSerializer"/> class.
/// </summary>
/// <param name="name">Name.</param>
public DefaultProtobufSerializer(string name)
{
_name = name;
}
/// <summary>
/// Gets the name.
/// </summary>
/// <value>The name.</value>
public string Name => _name;
/// <summary>
/// Deserialize the specified bytes.
/// </summary>
/// <returns>The deserialize.</returns>
/// <param name="bytes">Bytes.</param>
/// <typeparam name="T">The 1st type parameter.</typeparam>
public T Deserialize<T>(byte[] bytes)
{
using (MemoryStream ms = new MemoryStream(bytes))
{
return Serializer.Deserialize<T>(ms);
}
}
/// <summary>
/// Deserialize the specified bytes.
/// </summary>
/// <returns>The deserialize.</returns>
/// <param name="bytes">Bytes.</param>
/// <param name="type">Type.</param>
public object Deserialize(byte[] bytes, Type type)
{
using (MemoryStream ms = new MemoryStream(bytes))
{
return Serializer.Deserialize(type, ms);
}
throw new NotImplementedException();
}
/// <summary>
/// Serialize the specified value.
/// </summary>
/// <returns>The serialize.</returns>
/// <param name="value">Value.</param>
/// <typeparam name="T">The 1st type parameter.</typeparam>
public byte[] Serialize<T>(T value)
{
using (var ms = new MemoryStream())
{
Serializer.Serialize<T>(ms, value);
return ms.ToArray();
}
}
#region Mainly For Memcached
/// <summary>
/// Serializes the object.
/// </summary>
/// <returns>The object.</returns>
/// <param name="obj">Object.</param>
public ArraySegment<byte> SerializeObject(object obj)
{
using (var ms = new MemoryStream())
{
WriteType(ms, obj.GetType());
Serializer.NonGeneric.Serialize(ms, obj);
return new ArraySegment<byte>(ms.ToArray(), 0, (int)ms.Length);
}
}
/// <summary>
/// Deserializes the object.
/// </summary>
/// <returns>The object.</returns>
/// <param name="value">Value.</param>
public object DeserializeObject(ArraySegment<byte> value)
{
var raw = value.Array;
var count = value.Count;
var offset = value.Offset;
var type = ReadType(raw, ref offset, ref count);
using (var ms = new MemoryStream(raw, offset, count, writable: false))
{
return Serializer.NonGeneric.Deserialize(type, ms);
}
}
/// <summary>
/// Reads the type.
/// </summary>
/// <returns>The type.</returns>
/// <param name="buffer">Buffer.</param>
/// <param name="offset">Offset.</param>
/// <param name="count">Count.</param>
private Type ReadType(byte[] buffer, ref int offset, ref int count)
{
if (count < 4) throw new EndOfStreamException();
// len is size of header typeName(string)
var len = (int)buffer[offset++]
| (buffer[offset++] << 8)
| (buffer[offset++] << 16)
| (buffer[offset++] << 24);
count -= 4; // count is message total size, decr typeName length(int)
if (count < len) throw new EndOfStreamException();
var keyOffset = offset;
offset += len; // skip typeName body size
count -= len; // decr typeName body size
// avoid encode string
var key = new ArraySegment<byte>(buffer, keyOffset, len);
var typeName = Encoding.UTF8.GetString(key.Array, key.Offset, key.Count);
return Type.GetType(typeName, throwOnError: true);
}
/// <summary>
/// Writes the type.
/// </summary>
/// <param name="ms">Ms.</param>
/// <param name="type">Type.</param>
private void WriteType(MemoryStream ms, Type type)
{
var typeName = TypeHelper.BuildTypeName(type);
var typeArray = Encoding.UTF8.GetBytes(typeName);
var len = typeArray.Length;
// BinaryWrite Int32
ms.WriteByte((byte)len);
ms.WriteByte((byte)(len >> 8));
ms.WriteByte((byte)(len >> 16));
ms.WriteByte((byte)(len >> 24));
// BinaryWrite String
ms.Write(typeArray, 0, len);
}
#endregion
}
}