在C#中对GRPC的TLS支持

Zut*_*Zut 5 c# grpc

我是C#的新手,我需要在TLS上使用GRPC。

作为试运行,我正在修改主grpc存储库中提供的示例以使用TLS。为此,我发现了另一个SO问题,似乎是个好答案:如何为gRPC启用服务器端SSL?。但是,我得到一个错误Unhandled Exception: Grpc.Core.RpcException: Status(StatusCode=Unavailable, Detail="Connect Failed")(下面有完整的跟踪)。

如果我在原始的非tls代码中指定了错误的端口,或者只是不启动服务器,则会收到相同的错误。我在Ubuntu上使用dotnet core。

代码的重要部分在下面,也可以在github的fork中找到。

客户:

  var cacert = File.ReadAllText(@"../ca.crt");
  var clientcert = File.ReadAllText(@"../client.crt");
  var clientkey = File.ReadAllText(@"../client.key");
  var ssl = new SslCredentials(cacert, new KeyCertificatePair(clientcert, clientkey));
  var channel = new Channel("localhost", 555, ssl);
  var client = new Greeter.GreeterClient(channel);

  String user = "you";

  var reply = client.SayHello(new HelloRequest {Name = user});
  Console.WriteLine("Greeting: " + reply.Message);

  channel.ShutdownAsync().Wait();
  Console.WriteLine("Press any key to exit...");
  Console.ReadKey();
Run Code Online (Sandbox Code Playgroud)

服务器:

  var cacert = File.ReadAllText(@"../ca.crt");
  var servercert = File.ReadAllText(@"../server.crt");
  var serverkey = File.ReadAllText(@"../server.key");
  var keypair = new KeyCertificatePair(servercert, serverkey);
  var sslCredentials = new SslServerCredentials(new List<KeyCertificatePair>() {keypair}, cacert, false);

  var server = new Server
  {
    Services = {Greeter.BindService(new GreeterImpl())},
    Ports = {new ServerPort("0.0.0.0", 555, sslCredentials)}
  };
  server.Start();


  Console.WriteLine("Greeter server listening on port " + Port);
  Console.WriteLine("Press any key to stop the server...");
  Console.ReadKey();

  server.ShutdownAsync().Wait();
Run Code Online (Sandbox Code Playgroud)

程序的完整输出:

$ cd GreeterClient
$ dotnet run -f netcoreapp1.0

Unhandled Exception: Grpc.Core.RpcException: Status(StatusCode=Unavailable, Detail="Connect Failed")
   at System.Runtime.CompilerServices.TaskAwaiter.ThrowForNonSuccess(Task task)
   at System.Runtime.CompilerServices.TaskAwaiter.HandleNonSuccessAndDebuggerNotification(Task task)
   at Grpc.Core.Internal.AsyncCall`2.UnaryCall(TRequest msg)
   at Grpc.Core.DefaultCallInvoker.BlockingUnaryCall[TRequest,TResponse](Method`2 method, String host, CallOptions options, TRequest request)
   at Grpc.Core.Internal.InterceptingCallInvoker.BlockingUnaryCall[TRequest,TResponse](Method`2 method, String host, CallOptions options, TRequest request)
   at Helloworld.Greeter.GreeterClient.SayHello(HelloRequest request, CallOptions options)
   at Helloworld.Greeter.GreeterClient.SayHello(HelloRequest request, Metadata headers, Nullable`1 deadline, CancellationToken cancellationToken)
   at GreeterClient.Program.Main(String[] args)
Run Code Online (Sandbox Code Playgroud)

和服务器:

$ cd GreeterServer/
$ dotnet run -f netcoreapp1.0
Greeter server listening on port 50051
Press any key to stop the server...
Run Code Online (Sandbox Code Playgroud)

我是否犯了一些愚蠢的错误,还是在非Windows计算机上无法运行?有什么方法可以调试问题以弄清楚是怎么回事?

Pet*_*art 5

您的代码正确地实现了带有自签名证书的Mutual TLS-我能够在Windows netcore上运行它而无需做任何更改。

我确实需要稍微调整证书生成脚本:

  • 添加CLIENT-COMPUTERNAME用于客户端的环境变量定义/CN(通用名称)
  • 添加一个-config选项来消除unable to find 'distinguished_name' in config错误(在Windows上可能是openssl)

通过故意不匹配客户端连接到的主机(127.0.0.1而不是localhost),我能够重现与您相同的错误消息。因此,也许您只需要使用例如localhost客户端的通用名称来重新生成密钥。

  • 是的,您要连接的主机名需要与客户端通道目标匹配(因此,您可能需要重新生成证书以使其匹配)。通过设置env变量GRPC_VERBOSITY = debug启用更多日志记录,可以获取更多见解 (2认同)