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