117 lines
4.3 KiB
C#
117 lines
4.3 KiB
C#
using System;
|
|
using System.IO;
|
|
using System.Threading.Tasks;
|
|
using Google.Protobuf;
|
|
|
|
namespace SocketsSample.Protobuf
|
|
{
|
|
public class ProtobufInvocationAdapter : IInvocationAdapter
|
|
{
|
|
public async Task<InvocationDescriptor> CreateInvocationDescriptor(Stream stream, Func<string, Type[]> getParams)
|
|
{
|
|
return await Task.Run(() => CreateInvocationDescriptorInt(stream, getParams));
|
|
}
|
|
|
|
private static Task<InvocationDescriptor> CreateInvocationDescriptorInt(Stream stream, Func<string, Type[]> getParams)
|
|
{
|
|
var inputStream = new CodedInputStream(stream, leaveOpen: true);
|
|
var invocationHeader = new RpcInvocationHeader();
|
|
inputStream.ReadMessage(invocationHeader);
|
|
var argumentTypes = getParams(invocationHeader.Name);
|
|
|
|
var invocationDescriptor = new InvocationDescriptor();
|
|
invocationDescriptor.Method = invocationHeader.Name;
|
|
invocationDescriptor.Id = invocationHeader.Id.ToString();
|
|
invocationDescriptor.Arguments = new object[argumentTypes.Length];
|
|
|
|
var primitiveParser = PrimitiveValue.Parser;
|
|
|
|
for (var i = 0; i < argumentTypes.Length; i++)
|
|
{
|
|
var value = new PrimitiveValue();
|
|
inputStream.ReadMessage(value);
|
|
if (typeof(int) == argumentTypes[i])
|
|
{
|
|
invocationDescriptor.Arguments[i] = value.Int32Value;
|
|
}
|
|
else if (typeof(string) == argumentTypes[i])
|
|
{
|
|
invocationDescriptor.Arguments[i] = value.StringValue;
|
|
}
|
|
else
|
|
{
|
|
throw new InvalidOperationException();
|
|
}
|
|
}
|
|
|
|
return Task.FromResult(invocationDescriptor);
|
|
}
|
|
|
|
public async Task WriteInvocationResult(Stream stream, InvocationResultDescriptor resultDescriptor)
|
|
{
|
|
var outputStream = new CodedOutputStream(stream, leaveOpen: true);
|
|
outputStream.WriteMessage(new RpcMessageKind() { MessageKind = RpcMessageKind.Types.Kind.Result });
|
|
|
|
var resultHeader = new RpcInvocationResultHeader
|
|
{
|
|
Id = int.Parse(resultDescriptor.Id),
|
|
HasResult = resultDescriptor.Result != null
|
|
};
|
|
|
|
if (resultDescriptor.Error != null)
|
|
{
|
|
resultHeader.Error = resultDescriptor.Error;
|
|
}
|
|
|
|
outputStream.WriteMessage(resultHeader);
|
|
|
|
if (resultHeader.Error == null && resultDescriptor.Result != null)
|
|
{
|
|
var result = resultDescriptor.Result;
|
|
|
|
if (result.GetType() == typeof(int))
|
|
{
|
|
outputStream.WriteMessage(new PrimitiveValue { Int32Value = (int)result });
|
|
}
|
|
else if (result.GetType() == typeof(string))
|
|
{
|
|
outputStream.WriteMessage(new PrimitiveValue { StringValue = (string)result });
|
|
}
|
|
}
|
|
|
|
outputStream.Flush();
|
|
await stream.FlushAsync();
|
|
}
|
|
|
|
public async Task InvokeClientMethod(Stream stream, InvocationDescriptor invocationDescriptor)
|
|
{
|
|
var outputStream = new CodedOutputStream(stream, leaveOpen: true);
|
|
outputStream.WriteMessage(new RpcMessageKind() { MessageKind = RpcMessageKind.Types.Kind.Invocation });
|
|
|
|
var invocationHeader = new RpcInvocationHeader()
|
|
{
|
|
Id = 0,
|
|
Name = invocationDescriptor.Method,
|
|
NumArgs = invocationDescriptor.Arguments.Length
|
|
};
|
|
|
|
outputStream.WriteMessage(invocationHeader);
|
|
|
|
foreach (var arg in invocationDescriptor.Arguments)
|
|
{
|
|
if (arg.GetType() == typeof(int))
|
|
{
|
|
outputStream.WriteMessage(new PrimitiveValue { Int32Value = (int)arg });
|
|
}
|
|
else if (arg.GetType() == typeof(string))
|
|
{
|
|
outputStream.WriteMessage(new PrimitiveValue { StringValue = (string)arg });
|
|
}
|
|
}
|
|
|
|
outputStream.Flush();
|
|
await stream.FlushAsync();
|
|
}
|
|
}
|
|
}
|