相当于 WCF 服务发现的 gRPC

Jin*_*nov 6 .net c# wcf .net-core grpc

编辑:我不想使用 Consul 或 ZooKeeper。我想在本地网络上查找 Web 服务的地址。

什么是WCF,像服务发现类的GRPC等价物:ServiceDiscoveryBehaviorUdpDiscoveryEndpointDiscoveryClient此示例中使用:

服务:

using (ServiceHost serviceHost = new ServiceHost(typeof(CalculatorService), baseAddress))
{
    // Add calculator endpoint
    serviceHost.AddServiceEndpoint(typeof(ICalculator), new WSHttpBinding(), string.Empty);

    // ** DISCOVERY ** //
    // Make the service discoverable by adding the discovery behavior
    ServiceDiscoveryBehavior discoveryBehavior = new ServiceDiscoveryBehavior();
    serviceHost.Description.Behaviors.Add(discoveryBehavior);

    // Send announcements on UDP multicast transport
    discoveryBehavior.AnnouncementEndpoints.Add(new UdpAnnouncementEndpoint());

    // ** DISCOVERY ** //
    // Add the discovery endpoint that specifies where to publish the services
    serviceHost.Description.Endpoints.Add(new UdpDiscoveryEndpoint());

    // Open the ServiceHost to create listeners and start listening for messages.
    serviceHost.Open();
}
Run Code Online (Sandbox Code Playgroud)

客户:

{
    DiscoveryClient discoveryClient = new DiscoveryClient(new UdpDiscoveryEndpoint());

    Collection<EndpointDiscoveryMetadata> calculatorServices =   
        (Collection<EndpointDiscoveryMetadata>)
        discoveryClient.Find(new FindCriteria(typeof(ICalculator))).Endpoints;

    discoveryClient.Close();

    CalculatorClient client = new CalculatorClient();  
    client.Endpoint.Address = calculatorServices[0].Address;
}  
Run Code Online (Sandbox Code Playgroud)

Jin*_*nov 9

gRPC 没有相当于ServiceDiscoveryBehavior,UdpDiscoveryEndpoint或 的类DiscoveryClient

您必须System.Net.Sockets使用 UDP 广播编写自己的网络发现。

服务:

IPAddress ipAddress = Dns.GetHostEntry(Dns.GetHostName()).AddressList.
                      FirstOrDefault(ip => ip.AddressFamily == AddressFamily.InterNetwork);

Server grpcServer = new Server
{
    Services = { Simulator.BindService(new Service()) },
    Ports = { new ServerPort(ipAddress.ToString(), 8080, ServerCredentials.Insecure) }
};
grpcServer.Start();

Task.Run(() =>
{
    while (true)
    {
        UdpClient udpServer = new UdpClient(8888);
        IPEndPoint clientEndPoint = new IPEndPoint(IPAddress.Any, 0);
        byte[] clientRequestData = udpServer.Receive(ref clientEndPoint);
        string clientRequest = Encoding.ASCII.GetString(clientRequestData);
        Console.WriteLine($"Recived {clientRequest} from {clientEndPoint.Address}");

        byte[] responseData = Encoding.ASCII.GetBytes("Response");
        udpServer.Send(responseData, responseData.Length, clientEndPoint);
        udpServer.Close();
    }
});
Run Code Online (Sandbox Code Playgroud)

客户:

UdpClient udpClient = new UdpClient { EnableBroadcast = true };
byte[] requestData = Encoding.ASCII.GetBytes("Request");
udpClient.Send(requestData, requestData.Length, new IPEndPoint(IPAddress.Broadcast, 8888));

IPEndPoint serverEndPoint = new IPEndPoint(IPAddress.Any, 0);
byte[] serverResponseData = udpClient.Receive(ref serverEndPoint);
string serverResponse = Encoding.ASCII.GetString(serverResponseData);
Console.WriteLine($"Recived {serverResponse} from {serverEndPoint.Address}");

IPAddress ipAddress = serverEndPoint.Address;
udpClient.Close();

var grpcChannel = new Channel(ipAddress.ToString(), 8080, ChannelCredentials.Insecure);
var grpcClient = new Client(grpcChannel);
Run Code Online (Sandbox Code Playgroud)

  • 您能解释一下否决票吗?A) gRPC 库确实没有任何这样的类,B) 这段代码的作用与 WCF 代码的作用完全相同。 (3认同)