Skip to content
Snippets Groups Projects
Select Git revision
  • 8368b2e4b911a0e47cb2a2304513939ab34c74c3
  • master default protected
  • arm-aarch-platform
  • arm-platform
  • vjpai-patch-3
  • vjpai-patch-1
  • v1.27.x
  • jtattermusch-patch-2
  • jtattermusch-patch-1
  • update-java-worker-example-in-performance-docs
  • revert-21805-revert-21797-reintroduce_21527
  • revert-21804-tls-credentials-1
  • zhen_cleanup_namecheck
  • revert-21806-revert-21767-revert-21725-revert-21680-cq_ordering
  • vjpai-patch-2
  • revert-21766-tls-credentials-1
  • revert-21640-change_local_tcp_security_level
  • revert-21680-cq_ordering
  • revert-21527-unify_boringssl_deps2
  • revert-20803-grpclb_stabilization
  • fix-kokoro-rvm-key
  • v1.27.0
  • v1.27.0-pre2
  • v1.27.0-pre1
  • v1.26.0
  • v1.26.0-pre1
  • v1.25.0
  • v1.25.0-pre1
  • v1.24.3
  • v1.24.2
  • v1.24.1
  • v1.23.1
  • v1.24.0
  • v1.24.0-pre2
  • v1.24.0-pre1
  • v1.22.1
  • v1.23.0
  • v1.23.0-pre1
  • v1.22.0
  • v1.22.0-pre1
  • v1.21.4
41 results

ServerCallHandler.cs

Blame
  • user avatar
    Jan Tattermusch authored
    8368b2e4
    History
    ServerCallHandler.cs 12.77 KiB
    #region Copyright notice and license
    
    // Copyright 2015, Google Inc.
    // All rights reserved.
    //
    // Redistribution and use in source and binary forms, with or without
    // modification, are permitted provided that the following conditions are
    // met:
    //
    //     * Redistributions of source code must retain the above copyright
    // notice, this list of conditions and the following disclaimer.
    //     * Redistributions in binary form must reproduce the above
    // copyright notice, this list of conditions and the following disclaimer
    // in the documentation and/or other materials provided with the
    // distribution.
    //     * Neither the name of Google Inc. nor the names of its
    // contributors may be used to endorse or promote products derived from
    // this software without specific prior written permission.
    //
    // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
    // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
    // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
    // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
    // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
    // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
    // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
    // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
    // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
    // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
    // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
    
    #endregion
    
    using System;
    using System.Collections.Generic;
    using System.Linq;
    using System.Threading;
    using System.Threading.Tasks;
    using Grpc.Core.Internal;
    using Grpc.Core.Logging;
    using Grpc.Core.Utils;
    
    namespace Grpc.Core.Internal
    {
        internal interface IServerCallHandler
        {
            Task HandleCall(ServerRpcNew newRpc, GrpcEnvironment environment);
        }
    
        internal class UnaryServerCallHandler<TRequest, TResponse> : IServerCallHandler
            where TRequest : class
            where TResponse : class
        {
            static readonly ILogger Logger = GrpcEnvironment.Logger.ForType<UnaryServerCallHandler<TRequest, TResponse>>();
    
            readonly Method<TRequest, TResponse> method;
            readonly UnaryServerMethod<TRequest, TResponse> handler;
    
            public UnaryServerCallHandler(Method<TRequest, TResponse> method, UnaryServerMethod<TRequest, TResponse> handler)
            {
                this.method = method;
                this.handler = handler;
            }
    
            public async Task HandleCall(ServerRpcNew newRpc, GrpcEnvironment environment)
            {
                var asyncCall = new AsyncCallServer<TRequest, TResponse>(
                    method.ResponseMarshaller.Serializer,
                    method.RequestMarshaller.Deserializer,
                    environment);
    
                asyncCall.Initialize(newRpc.Call);
                var finishedTask = asyncCall.ServerSideCallAsync();
                var requestStream = new ServerRequestStream<TRequest, TResponse>(asyncCall);
                var responseStream = new ServerResponseStream<TRequest, TResponse>(asyncCall);
    
                Status status;
                var context = HandlerUtils.NewContext(newRpc, asyncCall.Peer, responseStream, asyncCall.CancellationToken);
                try
                {
                    Preconditions.CheckArgument(await requestStream.MoveNext());
                    var request = requestStream.Current;
                    // TODO(jtattermusch): we need to read the full stream so that native callhandle gets deallocated.
                    Preconditions.CheckArgument(!await requestStream.MoveNext());
                    var result = await handler(request, context);
                    status = context.Status;
                    await responseStream.WriteAsync(result);
                } 
                catch (Exception e)
                {
                    Logger.Error(e, "Exception occured in handler.");
                    status = HandlerUtils.StatusFromException(e);
                }
                try
                {
                    await responseStream.WriteStatusAsync(status, context.ResponseTrailers);
                }
                catch (OperationCanceledException)
                {
                    // Call has been already cancelled.
                }
                await finishedTask;
            }
        }
    
        internal class ServerStreamingServerCallHandler<TRequest, TResponse> : IServerCallHandler
            where TRequest : class
            where TResponse : class
        {
            static readonly ILogger Logger = GrpcEnvironment.Logger.ForType<ServerStreamingServerCallHandler<TRequest, TResponse>>();
    
            readonly Method<TRequest, TResponse> method;
            readonly ServerStreamingServerMethod<TRequest, TResponse> handler;
    
            public ServerStreamingServerCallHandler(Method<TRequest, TResponse> method, ServerStreamingServerMethod<TRequest, TResponse> handler)
            {
                this.method = method;
                this.handler = handler;
            }
    
            public async Task HandleCall(ServerRpcNew newRpc, GrpcEnvironment environment)
            {
                var asyncCall = new AsyncCallServer<TRequest, TResponse>(
                    method.ResponseMarshaller.Serializer,
                    method.RequestMarshaller.Deserializer,
                    environment);
    
                asyncCall.Initialize(newRpc.Call);
                var finishedTask = asyncCall.ServerSideCallAsync();
                var requestStream = new ServerRequestStream<TRequest, TResponse>(asyncCall);
                var responseStream = new ServerResponseStream<TRequest, TResponse>(asyncCall);
    
                Status status;
                var context = HandlerUtils.NewContext(newRpc, asyncCall.Peer, responseStream, asyncCall.CancellationToken);
                try
                {
                    Preconditions.CheckArgument(await requestStream.MoveNext());
                    var request = requestStream.Current;
                    // TODO(jtattermusch): we need to read the full stream so that native callhandle gets deallocated.
                    Preconditions.CheckArgument(!await requestStream.MoveNext());
                    await handler(request, responseStream, context);
                    status = context.Status;
                }
                catch (Exception e)
                {
                    Logger.Error(e, "Exception occured in handler.");
                    status = HandlerUtils.StatusFromException(e);
                }
    
                try
                {
                    await responseStream.WriteStatusAsync(status, context.ResponseTrailers);
                }
                catch (OperationCanceledException)
                {
                    // Call has been already cancelled.
                }
                await finishedTask;
            }
        }
    
        internal class ClientStreamingServerCallHandler<TRequest, TResponse> : IServerCallHandler
            where TRequest : class
            where TResponse : class
        {
            static readonly ILogger Logger = GrpcEnvironment.Logger.ForType<ClientStreamingServerCallHandler<TRequest, TResponse>>();
    
            readonly Method<TRequest, TResponse> method;
            readonly ClientStreamingServerMethod<TRequest, TResponse> handler;
    
            public ClientStreamingServerCallHandler(Method<TRequest, TResponse> method, ClientStreamingServerMethod<TRequest, TResponse> handler)
            {
                this.method = method;
                this.handler = handler;
            }
    
            public async Task HandleCall(ServerRpcNew newRpc, GrpcEnvironment environment)
            {
                var asyncCall = new AsyncCallServer<TRequest, TResponse>(
                    method.ResponseMarshaller.Serializer,
                    method.RequestMarshaller.Deserializer,
                    environment);
    
                asyncCall.Initialize(newRpc.Call);
                var finishedTask = asyncCall.ServerSideCallAsync();
                var requestStream = new ServerRequestStream<TRequest, TResponse>(asyncCall);
                var responseStream = new ServerResponseStream<TRequest, TResponse>(asyncCall);
    
                Status status;
                var context = HandlerUtils.NewContext(newRpc, asyncCall.Peer, responseStream, asyncCall.CancellationToken);
                try
                {
                    var result = await handler(requestStream, context);
                    status = context.Status;
                    try
                    {
                        await responseStream.WriteAsync(result);
                    }
                    catch (OperationCanceledException)
                    {
                        status = Status.DefaultCancelled;
                    }
                }
                catch (Exception e)
                {
                    Logger.Error(e, "Exception occured in handler.");
                    status = HandlerUtils.StatusFromException(e);
                }
    
                try
                {
                    await responseStream.WriteStatusAsync(status, context.ResponseTrailers);
                }
                catch (OperationCanceledException)
                {
                    // Call has been already cancelled.
                }
                await finishedTask;
            }
        }
    
        internal class DuplexStreamingServerCallHandler<TRequest, TResponse> : IServerCallHandler
            where TRequest : class
            where TResponse : class
        {
            static readonly ILogger Logger = GrpcEnvironment.Logger.ForType<DuplexStreamingServerCallHandler<TRequest, TResponse>>();
    
            readonly Method<TRequest, TResponse> method;
            readonly DuplexStreamingServerMethod<TRequest, TResponse> handler;
    
            public DuplexStreamingServerCallHandler(Method<TRequest, TResponse> method, DuplexStreamingServerMethod<TRequest, TResponse> handler)
            {
                this.method = method;
                this.handler = handler;
            }
    
            public async Task HandleCall(ServerRpcNew newRpc, GrpcEnvironment environment)
            {
                var asyncCall = new AsyncCallServer<TRequest, TResponse>(
                    method.ResponseMarshaller.Serializer,
                    method.RequestMarshaller.Deserializer,
                    environment);
    
                asyncCall.Initialize(newRpc.Call);
                var finishedTask = asyncCall.ServerSideCallAsync();
                var requestStream = new ServerRequestStream<TRequest, TResponse>(asyncCall);
                var responseStream = new ServerResponseStream<TRequest, TResponse>(asyncCall);
    
                Status status;
                var context = HandlerUtils.NewContext(newRpc, asyncCall.Peer, responseStream, asyncCall.CancellationToken);
                try
                {
                    await handler(requestStream, responseStream, context);
                    status = context.Status;
                }
                catch (Exception e)
                {
                    Logger.Error(e, "Exception occured in handler.");
                    status = HandlerUtils.StatusFromException(e);
                }
                try
                {
                    await responseStream.WriteStatusAsync(status, context.ResponseTrailers);
                }
                catch (OperationCanceledException)
                {
                    // Call has been already cancelled.
                }
                await finishedTask;
            }
        }
    
        internal class NoSuchMethodCallHandler : IServerCallHandler
        {
            public static readonly NoSuchMethodCallHandler Instance = new NoSuchMethodCallHandler();
    
            public async Task HandleCall(ServerRpcNew newRpc, GrpcEnvironment environment)
            {
                // We don't care about the payload type here.
                var asyncCall = new AsyncCallServer<byte[], byte[]>(
                    (payload) => payload, (payload) => payload, environment);
                
                asyncCall.Initialize(newRpc.Call);
                var finishedTask = asyncCall.ServerSideCallAsync();
                var responseStream = new ServerResponseStream<byte[], byte[]>(asyncCall);
    
                await responseStream.WriteStatusAsync(new Status(StatusCode.Unimplemented, "No such method."), Metadata.Empty);
                await finishedTask;
            }
        }
    
        internal static class HandlerUtils
        {
            public static Status StatusFromException(Exception e)
            {
                var rpcException = e as RpcException;
                if (rpcException != null)
                {
                    // use the status thrown by handler.
                    return rpcException.Status;
                }
    
                // TODO(jtattermusch): what is the right status code here?
                return new Status(StatusCode.Unknown, "Exception was thrown by handler.");
            }
    
            public static ServerCallContext NewContext<TRequest, TResponse>(ServerRpcNew newRpc, string peer, ServerResponseStream<TRequest, TResponse> serverResponseStream, CancellationToken cancellationToken)
                where TRequest : class
                where TResponse : class
            {
                DateTime realtimeDeadline = newRpc.Deadline.ToClockType(GPRClockType.Realtime).ToDateTime();
    
                return new ServerCallContext(
                    newRpc.Method, newRpc.Host, peer, realtimeDeadline,
                    newRpc.RequestMetadata, cancellationToken, serverResponseStream.WriteResponseHeadersAsync, serverResponseStream);
            }
        }
    }