mirror of https://github.com/grpc/grpc.git
Merge pull request #16367 from jtattermusch/csharp_new_serialization_api
Add new C# serialization APIpull/16707/head
commit
d4356bf719
6 changed files with 420 additions and 12 deletions
@ -0,0 +1,119 @@ |
|||||||
|
#region Copyright notice and license |
||||||
|
|
||||||
|
// Copyright 2018 The gRPC Authors |
||||||
|
// |
||||||
|
// Licensed under the Apache License, Version 2.0 (the "License"); |
||||||
|
// you may not use this file except in compliance with the License. |
||||||
|
// You may obtain a copy of the License at |
||||||
|
// |
||||||
|
// http://www.apache.org/licenses/LICENSE-2.0 |
||||||
|
// |
||||||
|
// Unless required by applicable law or agreed to in writing, software |
||||||
|
// distributed under the License is distributed on an "AS IS" BASIS, |
||||||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
||||||
|
// See the License for the specific language governing permissions and |
||||||
|
// limitations under the License. |
||||||
|
|
||||||
|
#endregion |
||||||
|
|
||||||
|
using System; |
||||||
|
using System.Collections.Generic; |
||||||
|
using System.Diagnostics; |
||||||
|
using System.IO; |
||||||
|
using System.Linq; |
||||||
|
using System.Threading; |
||||||
|
using System.Threading.Tasks; |
||||||
|
|
||||||
|
using Grpc.Core; |
||||||
|
using Grpc.Core.Internal; |
||||||
|
using Grpc.Core.Utils; |
||||||
|
using NUnit.Framework; |
||||||
|
|
||||||
|
namespace Grpc.Core.Tests |
||||||
|
{ |
||||||
|
public class ContextualMarshallerTest |
||||||
|
{ |
||||||
|
const string Host = "127.0.0.1"; |
||||||
|
|
||||||
|
MockServiceHelper helper; |
||||||
|
Server server; |
||||||
|
Channel channel; |
||||||
|
|
||||||
|
[SetUp] |
||||||
|
public void Init() |
||||||
|
{ |
||||||
|
var contextualMarshaller = new Marshaller<string>( |
||||||
|
(str, serializationContext) => |
||||||
|
{ |
||||||
|
if (str == "UNSERIALIZABLE_VALUE") |
||||||
|
{ |
||||||
|
// Google.Protobuf throws exception inherited from IOException |
||||||
|
throw new IOException("Error serializing the message."); |
||||||
|
} |
||||||
|
if (str == "SERIALIZE_TO_NULL") |
||||||
|
{ |
||||||
|
return; |
||||||
|
} |
||||||
|
var bytes = System.Text.Encoding.UTF8.GetBytes(str); |
||||||
|
serializationContext.Complete(bytes); |
||||||
|
}, |
||||||
|
(deserializationContext) => |
||||||
|
{ |
||||||
|
var buffer = deserializationContext.PayloadAsNewBuffer(); |
||||||
|
Assert.AreEqual(buffer.Length, deserializationContext.PayloadLength); |
||||||
|
var s = System.Text.Encoding.UTF8.GetString(buffer); |
||||||
|
if (s == "UNPARSEABLE_VALUE") |
||||||
|
{ |
||||||
|
// Google.Protobuf throws exception inherited from IOException |
||||||
|
throw new IOException("Error parsing the message."); |
||||||
|
} |
||||||
|
return s; |
||||||
|
}); |
||||||
|
helper = new MockServiceHelper(Host, contextualMarshaller); |
||||||
|
server = helper.GetServer(); |
||||||
|
server.Start(); |
||||||
|
channel = helper.GetChannel(); |
||||||
|
} |
||||||
|
|
||||||
|
[TearDown] |
||||||
|
public void Cleanup() |
||||||
|
{ |
||||||
|
channel.ShutdownAsync().Wait(); |
||||||
|
server.ShutdownAsync().Wait(); |
||||||
|
} |
||||||
|
|
||||||
|
[Test] |
||||||
|
public void UnaryCall() |
||||||
|
{ |
||||||
|
helper.UnaryHandler = new UnaryServerMethod<string, string>((request, context) => |
||||||
|
{ |
||||||
|
return Task.FromResult(request); |
||||||
|
}); |
||||||
|
Assert.AreEqual("ABC", Calls.BlockingUnaryCall(helper.CreateUnaryCall(), "ABC")); |
||||||
|
} |
||||||
|
|
||||||
|
[Test] |
||||||
|
public void ResponseParsingError_UnaryResponse() |
||||||
|
{ |
||||||
|
helper.UnaryHandler = new UnaryServerMethod<string, string>((request, context) => |
||||||
|
{ |
||||||
|
return Task.FromResult("UNPARSEABLE_VALUE"); |
||||||
|
}); |
||||||
|
|
||||||
|
var ex = Assert.Throws<RpcException>(() => Calls.BlockingUnaryCall(helper.CreateUnaryCall(), "REQUEST")); |
||||||
|
Assert.AreEqual(StatusCode.Internal, ex.Status.StatusCode); |
||||||
|
} |
||||||
|
|
||||||
|
[Test] |
||||||
|
public void RequestSerializationError_BlockingUnary() |
||||||
|
{ |
||||||
|
Assert.Throws<IOException>(() => Calls.BlockingUnaryCall(helper.CreateUnaryCall(), "UNSERIALIZABLE_VALUE")); |
||||||
|
} |
||||||
|
|
||||||
|
[Test] |
||||||
|
public void SerializationResultIsNull_BlockingUnary() |
||||||
|
{ |
||||||
|
Assert.Throws<NullReferenceException>(() => Calls.BlockingUnaryCall(helper.CreateUnaryCall(), "SERIALIZE_TO_NULL")); |
||||||
|
} |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,105 @@ |
|||||||
|
#region Copyright notice and license |
||||||
|
|
||||||
|
// Copyright 2018 The gRPC Authors |
||||||
|
// |
||||||
|
// Licensed under the Apache License, Version 2.0 (the "License"); |
||||||
|
// you may not use this file except in compliance with the License. |
||||||
|
// You may obtain a copy of the License at |
||||||
|
// |
||||||
|
// http://www.apache.org/licenses/LICENSE-2.0 |
||||||
|
// |
||||||
|
// Unless required by applicable law or agreed to in writing, software |
||||||
|
// distributed under the License is distributed on an "AS IS" BASIS, |
||||||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
||||||
|
// See the License for the specific language governing permissions and |
||||||
|
// limitations under the License. |
||||||
|
|
||||||
|
#endregion |
||||||
|
|
||||||
|
using System; |
||||||
|
using System.Collections.Generic; |
||||||
|
using System.Diagnostics; |
||||||
|
using System.IO; |
||||||
|
using System.Linq; |
||||||
|
using System.Threading; |
||||||
|
using System.Threading.Tasks; |
||||||
|
|
||||||
|
using Grpc.Core; |
||||||
|
using Grpc.Core.Internal; |
||||||
|
using Grpc.Core.Utils; |
||||||
|
using NUnit.Framework; |
||||||
|
|
||||||
|
namespace Grpc.Core.Tests |
||||||
|
{ |
||||||
|
public class MarshallerTest |
||||||
|
{ |
||||||
|
[Test] |
||||||
|
public void ContextualSerializerEmulation() |
||||||
|
{ |
||||||
|
Func<string, byte[]> simpleSerializer = System.Text.Encoding.UTF8.GetBytes; |
||||||
|
Func<byte[], string> simpleDeserializer = System.Text.Encoding.UTF8.GetString; |
||||||
|
var marshaller = new Marshaller<string>(simpleSerializer, |
||||||
|
simpleDeserializer); |
||||||
|
|
||||||
|
Assert.AreSame(simpleSerializer, marshaller.Serializer); |
||||||
|
Assert.AreSame(simpleDeserializer, marshaller.Deserializer); |
||||||
|
|
||||||
|
// test that emulated contextual serializer and deserializer work |
||||||
|
string origMsg = "abc"; |
||||||
|
var serializationContext = new FakeSerializationContext(); |
||||||
|
marshaller.ContextualSerializer(origMsg, serializationContext); |
||||||
|
|
||||||
|
var deserializationContext = new FakeDeserializationContext(serializationContext.Payload); |
||||||
|
Assert.AreEqual(origMsg, marshaller.ContextualDeserializer(deserializationContext)); |
||||||
|
} |
||||||
|
|
||||||
|
[Test] |
||||||
|
public void SimpleSerializerEmulation() |
||||||
|
{ |
||||||
|
Action<string, SerializationContext> contextualSerializer = (str, context) => |
||||||
|
{ |
||||||
|
var bytes = System.Text.Encoding.UTF8.GetBytes(str); |
||||||
|
context.Complete(bytes); |
||||||
|
}; |
||||||
|
Func<DeserializationContext, string> contextualDeserializer = (context) => |
||||||
|
{ |
||||||
|
return System.Text.Encoding.UTF8.GetString(context.PayloadAsNewBuffer()); |
||||||
|
}; |
||||||
|
var marshaller = new Marshaller<string>(contextualSerializer, contextualDeserializer); |
||||||
|
|
||||||
|
Assert.AreSame(contextualSerializer, marshaller.ContextualSerializer); |
||||||
|
Assert.AreSame(contextualDeserializer, marshaller.ContextualDeserializer); |
||||||
|
|
||||||
|
// test that emulated serializer and deserializer work |
||||||
|
var origMsg = "abc"; |
||||||
|
var serialized = marshaller.Serializer(origMsg); |
||||||
|
Assert.AreEqual(origMsg, marshaller.Deserializer(serialized)); |
||||||
|
} |
||||||
|
|
||||||
|
class FakeSerializationContext : SerializationContext |
||||||
|
{ |
||||||
|
public byte[] Payload; |
||||||
|
public override void Complete(byte[] payload) |
||||||
|
{ |
||||||
|
this.Payload = payload; |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
class FakeDeserializationContext : DeserializationContext |
||||||
|
{ |
||||||
|
public byte[] payload; |
||||||
|
|
||||||
|
public FakeDeserializationContext(byte[] payload) |
||||||
|
{ |
||||||
|
this.payload = payload; |
||||||
|
} |
||||||
|
|
||||||
|
public override int PayloadLength => payload.Length; |
||||||
|
|
||||||
|
public override byte[] PayloadAsNewBuffer() |
||||||
|
{ |
||||||
|
return payload; |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,46 @@ |
|||||||
|
#region Copyright notice and license |
||||||
|
|
||||||
|
// Copyright 2018 The gRPC Authors |
||||||
|
// |
||||||
|
// Licensed under the Apache License, Version 2.0 (the "License"); |
||||||
|
// you may not use this file except in compliance with the License. |
||||||
|
// You may obtain a copy of the License at |
||||||
|
// |
||||||
|
// http://www.apache.org/licenses/LICENSE-2.0 |
||||||
|
// |
||||||
|
// Unless required by applicable law or agreed to in writing, software |
||||||
|
// distributed under the License is distributed on an "AS IS" BASIS, |
||||||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
||||||
|
// See the License for the specific language governing permissions and |
||||||
|
// limitations under the License. |
||||||
|
|
||||||
|
#endregion |
||||||
|
|
||||||
|
namespace Grpc.Core |
||||||
|
{ |
||||||
|
/// <summary> |
||||||
|
/// Provides access to the payload being deserialized when deserializing messages. |
||||||
|
/// </summary> |
||||||
|
public abstract class DeserializationContext |
||||||
|
{ |
||||||
|
/// <summary> |
||||||
|
/// Get the total length of the payload in bytes. |
||||||
|
/// </summary> |
||||||
|
public abstract int PayloadLength { get; } |
||||||
|
|
||||||
|
/// <summary> |
||||||
|
/// Gets the entire payload as a newly allocated byte array. |
||||||
|
/// Once the byte array is returned, the byte array becomes owned by the caller and won't be ever accessed or reused by gRPC again. |
||||||
|
/// NOTE: Obtaining the buffer as a newly allocated byte array is the simplest way of accessing the payload, |
||||||
|
/// but it can have important consequences in high-performance scenarios. |
||||||
|
/// In particular, using this method usually requires copying of the entire buffer one extra time. |
||||||
|
/// Also, allocating a new buffer each time can put excessive pressure on GC, especially if |
||||||
|
/// the payload is more than 86700 bytes large (which means the newly allocated buffer will be placed in LOH, |
||||||
|
/// and LOH object can only be garbage collected via a full ("stop the world") GC run). |
||||||
|
/// NOTE: Deserializers are expected not to call this method more than once per received message |
||||||
|
/// (as there is no practical reason for doing so) and <c>DeserializationContext</c> implementations are free to assume so. |
||||||
|
/// </summary> |
||||||
|
/// <returns>byte array containing the entire payload.</returns> |
||||||
|
public abstract byte[] PayloadAsNewBuffer(); |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,34 @@ |
|||||||
|
#region Copyright notice and license |
||||||
|
|
||||||
|
// Copyright 2018 The gRPC Authors |
||||||
|
// |
||||||
|
// Licensed under the Apache License, Version 2.0 (the "License"); |
||||||
|
// you may not use this file except in compliance with the License. |
||||||
|
// You may obtain a copy of the License at |
||||||
|
// |
||||||
|
// http://www.apache.org/licenses/LICENSE-2.0 |
||||||
|
// |
||||||
|
// Unless required by applicable law or agreed to in writing, software |
||||||
|
// distributed under the License is distributed on an "AS IS" BASIS, |
||||||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
||||||
|
// See the License for the specific language governing permissions and |
||||||
|
// limitations under the License. |
||||||
|
|
||||||
|
#endregion |
||||||
|
|
||||||
|
namespace Grpc.Core |
||||||
|
{ |
||||||
|
/// <summary> |
||||||
|
/// Provides storage for payload when serializing a message. |
||||||
|
/// </summary> |
||||||
|
public abstract class SerializationContext |
||||||
|
{ |
||||||
|
/// <summary> |
||||||
|
/// Use the byte array as serialized form of current message and mark serialization process as complete. |
||||||
|
/// Complete() can only be called once. By calling this method the caller gives up the ownership of the |
||||||
|
/// payload which must not be accessed afterwards. |
||||||
|
/// </summary> |
||||||
|
/// <param name="payload">the serialized form of current message</param> |
||||||
|
public abstract void Complete(byte[] payload); |
||||||
|
} |
||||||
|
} |
Loading…
Reference in new issue