How should a GRPC Service be hosted?

后端 未结 4 1745
予麋鹿
予麋鹿 2020-12-14 02:49

I have created a GRPC Server in C# using the example given at Link. Now I want to figure out as how should I be hosting this server so that I achieve following:

相关标签:
4条回答
  • 2020-12-14 03:32

    We can use Microsoft.Extensions.Hosting pacakge to host a .net core console application by using the HostBuilder API to start building gRPC host and setting it up.

    In order to run the gRPC service, we first need to start/stop Grpc.Core.Server in a hosted service. A hosted service is basically a piece of code that is run by the host when the host itself is started and the same for when it is stopped. The following code implement a GrpcHostedService to override IHostedService interface:

    using System.Threading;
    using System.Threading.Tasks;
    using Grpc.Core;
    using Microsoft.Extensions.Hosting;
    
    namespace Grpc.Host
    {
        public class GrpcHostedService: IHostedService
        {
            private Server _server;
    
            public GrpcHostedService(Server server)
            {
                _server = server;
            }
    
            public Task StartAsync(CancellationToken cancellationToken)
            {
                _server.Start();
                return Task.CompletedTask;
            }
    
            public async Task StopAsync(CancellationToken cancellationToken) => await _server.ShutdownAsync();
        }
    }
    

    In the Program.cs, use HostBuilder API to start building our grpc host and setting it up:

    public class Program
    {
        public static async Task Main(string[] args)
        {
            var hostBuilder = new HostBuilder()
                 // Add configuration, logging, ...
                .ConfigureServices((hostContext, services) =>
                {
                    // Better to use Dependency Injection for GreeterImpl
                    Server server = new Server
                    {
                        Services = {Greeter.BindService(new GreeterImpl())},
                        Ports = {new ServerPort("localhost", 5000, ServerCredentials.Insecure)}
                    };
                    services.AddSingleton<Server>(server);
                    services.AddSingleton<IHostedService, GrpcHostedService>();
                });
    
            await hostBuilder.RunConsoleAsync();
        }
    }
    

    By doing this, the generic host will automatically run StartAsync on our hosted service, which in turn will call StartAsync on the Server instance, essentially start the gRPC server.

    When we shut down the host with Control-C, the generic host will automatically call StopAsync on our hosted service, which again will call StopAsync on the Server instance which will do some clean up.

    For other configuration in HostBuilder, you can see this blog.

    0 讨论(0)
  • 2020-12-14 03:32

    Currently gRPC doesn't support integration with ASP.Net/IIS. You would need to host the server in a console or as a Windows service.

    Likely you would want this to be a Windows service to make it easier to keep the server running across reboots or crashes. If you want to easily turn your console application into a Windows service I would recommend using the excellent TopShelf Nuget.

    Updating the service can be done as you would a console app.

    • Stop the Windows service. net stop <service-name}>
    • Copy the updated assemblies.
    • Start the Windowsservice net start <service-name>
    0 讨论(0)
  • 2020-12-14 03:49

    I'm going to add one more option.

    With dot net core, you can run this as a Linux Daemon now.

    0 讨论(0)
  • 2020-12-14 03:52

    My company (Shortbar) is building the application server for a hotel management system called HOLMS on gRPC. Our setup is as follows:

    • HOLMS.Application is a .NET class library (assembly) that does the actual work of the server
    • HOLMS.Application.ConsoleRunner is a C# console application that hosts HOLMS.Application. The console runner is used by (1) developers for convenience (mentioned in the question) as well as (2) production scenarios running inside a Docker container, where the container runtime (e.g. Amazon ECS) implements job control/scaling. It follows "12 factor app" guidelines, including running itself as a single, standalone, stateless process, fast startup/shutdown, and environment-variable config injection. The system logs to stdout which gets drained however stdout is drained in the prod environment (e.g. Sumo, logstash, etc). This is how our SaaS multi-tenant solution will go into production.
    • HOLMS.Application.ServiceRunner packages HOLMS.Application into a Windows service, for more traditional, on-premise situations where a customer's IT group will run the service themselves. This package uses the Windows registry for configuration and relies on Windows service job control for startup/shutdown/restarts. It logs to the Windows Event Log.

    The ConsoleRunner and ServiceRunner each are only about 200 lines of code; for the most part, they just wrap the Application package, and call into it.

    Hope this helps.

    0 讨论(0)
提交回复
热议问题