gRPC services communicating through the Istio mesh

gRPC services communicating through the Istio mesh

[[433796]]

introduction

This article verifies the communication process of gRCP service consumer mesha and gRPC service provider meshb deployed in the Istio mesh. This example makes it easier to connect an external registry to the mesh.

1. Isito configuration tips

Isito has been installed in the Kubernetes cluster, and the following parameters are checked to see if they are set correctly.

istio-config.yaml content

  1. ---  
  2. apiVersion: install.istio.io/v1alpha1
  3. kind: IstioOperator
  4. spec:
  5. profile: default  
  6. values ​​:
  7. global :
  8. logging:
  9. level : default :debug
  10. meshConfig:
  11. accessLogFile: /dev/stdout
  12. defaultConfig:
  13. holdApplicationUntilProxyStarts: true  
  14. proxyMetadata:
  15. ISTIO_META_DNS_CAPTURE: "true"  
  16. ISTIO_META_DNS_AUTO_ALLOCATE: "true"  
  17. components:
  18. Pilot:
  19. hub: istio
  20. tag: 1.10.4

Key Parameters

parameter illustrate
holdApplicationUntilProxyStarts The default value is false. If set to true, the business container must be started after the sidecar proxy container is started.
ISTIO_META_DNS_CAPTURE The default value is false. Setting it to true means turning on the DNS proxy, and DNS requests will be forwarded to the sidecar.
ISTIO_META_DNS_AUTO_ALLOCATE The default value is false. If set to true, the DNS proxy will automatically assign IP addresses to ServiceEntrys, and no IP address needs to be specified.

Execute the following command to take effect

  1. istioctl install -y -f istio-config.yaml

Verify the configuration after it takes effect by running the following command

  1. kubectl describe IstioOperator installed-state -n istio-system >> istioOperator.conf

2. Example

Defining Proto

The client and server communicate through the simple method SayHello.

  1. syntax = "proto3" ;
  2.  
  3. option java_multiple_files = true ;
  4. option java_package = "com.melon.test.client.grpc" ;
  5. option java_outer_classname = "HelloMesh" ;
  6.  
  7. package meshgrpc;
  8.  
  9. service Mesher {
  10.  
  11. rpc SayHello (HelloRequest) returns (HelloReply) {}
  12. }
  13.  
  14. message HelloRequest {
  15. string name = 1;
  16. }
  17.  
  18. message HelloReply {
  19. string message = 1;
  20. }

Service consumer mesha

  1. @RestController
  2. public class MeshSender {
  3.  
  4. @GetMapping( "demo" )
  5. public String meshWorker(){
  6.  
  7. String target = "dns:///AppMeshClient.mesh:50000" ;
  8. // String target = "127.0.0.1:50000" ;
  9. ManagedChannel channel = ManagedChannelBuilder.forTarget(target)
  10. .usePlaintext()
  11. .build();
  12. MesherGrpc.MesherBlockingStub blockingStub = MesherGrpc.newBlockingStub(channel);
  13. HelloRequest request = HelloRequest.newBuilder().setName( "mesh demo!" ).build();
  14. HelloReply reply = blockingStub.sayHello(request);
  15. System. out .println(reply.getMessage());
  16. return reply.getMessage();
  17.  
  18. }
  19. }

Service provider meshb

  1. @Component
  2. public class MeshReceiver {
  3.  
  4.  
  5. @PostConstruct
  6. public void receiverWorker() throws IOException {
  7. int port = 50000;
  8. Server server = ServerBuilder.forPort(port)
  9. .addService(new MeshBService())
  10. .build()
  11. .start();
  12. System. out .println( "Server started." );
  13. }
  14.  
  15. class MeshBService extends MesherGrpc.MesherImplBase {
  16.  
  17. public void sayHello(HelloRequest request,
  18. io.grpc.stub.StreamObserver<HelloReply> responseObserver) {
  19.  
  20. System. out .println( "receiver client message: " + request.getName());
  21. HelloReply reply = HelloReply.newBuilder().setMessage( "I'm from server " + request.getName()).build();
  22. responseObserver.onNext(reply);
  23. responseObserver.onCompleted();
  24. }
  25. }
  26.  
  27. }

Image push

  1. <plugin>
  2. <groupId>com.google.cloud.tools</groupId>
  3. <artifactId>jib-maven-plugin</artifactId>
  4. <version>3.1.4</version>
  5. <configuration>
  6. < from >
  7. <image>
  8. harbor.xx/x/java:8u212- full  
  9. </image>
  10. <auth>
  11. <username>${harbor.username}</username>
  12. < password >${harbor. password }</ password >
  13. </auth>
  14. </ from >
  15. < to >
  16. <image>xxx/x/${project. name }</image>
  17. <auth>
  18. <username>${harbor.username}</username>
  19. < password >${harbor. password }</ password >
  20. </auth>
  21. <tags>
  22. <tag>${project.version}</tag>
  23. <tag>latest</tag>
  24. </tags>
  25. </ to >
  26. <container>
  27. <ports>
  28. <port>x</port>
  29. </ports>
  30. <creationTime>USE_CURRENT_TIMESTAMP</creationTime>
  31. </container>
  32. </configuration>
  33. </plugin>

Note: Use the Maven plugin jib to execute the following command "mvn compile jib:build" to push the image to the harbor repository

3. gRPC Service Provider Deployment

Set up the Deployment of the meshb service

  1. ---  
  2. apiVersion: apps/v1
  3. kind: Deployment
  4. metadata:
  5. name : meshb
  6. labels:
  7. app: meshb
  8. spec:
  9. selector:
  10. matchLabels:
  11. app: meshb
  12. replicas: 1
  13. template:
  14. metadata:
  15. labels:
  16. app: meshb
  17. spec:
  18. imagePullSecrets:
  19. - name : xxxx #warehouse name
  20. containers:
  21. - name : meshb
  22. image: xxxx/x/meshb:latest
  23. imagePullPolicy: Always
  24. ports:
  25. - containerPort: 50000

Execute the following command to take effect

  1. kubectl apply -f meshb.yaml
  2. deployment.apps/meshb created

Check the running status is normal

  1. # kubectl get pods
  2. NAME READY STATUS RESTARTS AGE
  3. meshb-565945d794-wcb8z 2/2 Running 0 9m19s

Check the startup log to see if the startup is successful

  1. # kubectl logs meshb-565945d794-wcb8z -n default  
  2.  
  3. Server started.
  4. 2021-11-05 09:21:19.828 INFO 1 --- [main] osbwembedded.tomcat.TomcatWebServer: Tomcat started on port(s): 8181 (http) with context path ''  
  5. 2021-11-05 09:21:19.847 INFO 1 --- [main] com.melon.test.MeshbApplication : Started MeshbApplication in 3.859 seconds (JVM running for 4.344)  

Note: At this point, the gRPC service provider has been deployed.

4. gRPC Service Consumer Deployment

Set up the Deployment of the mesha service

  1. ---  
  2. apiVersion: apps/v1
  3. kind: Deployment
  4. metadata:
  5. name : mesha
  6. labels:
  7. app: mesha
  8. spec:
  9. selector:
  10. matchLabels:
  11. app: mesha
  12. replicas: 1
  13. template:
  14. metadata:
  15. labels:
  16. app: mesha
  17. spec:
  18. imagePullSecrets:
  19. - name : middleware
  20. containers:
  21. - name : mesha
  22. image: harbor.hellobike.cn/base/mesha:latest
  23. ports:
  24. - containerPort: 7171
  25. imagePullPolicy: Always
  26. ---  
  27. apiVersion: v1
  28. kind: Service
  29. metadata:
  30. name : mesha
  31. spec:
  32. selector:
  33. app: mesha
  34. type: LoadBalancer
  35. ports:
  36. - name : web

Execute command deployment

  1. # kubectl apply -f mesha.yaml
  2.  
  3. deployment.apps/mesha created
  4.  
  5. service/mesha created

View service status

  1. # kubectl get service
  2. NAME TYPE CLUSTER-IP EXTERNAL-IP PORT(S) AGE
  3. mesha LoadBalancer 10.x.61.x <pending> 7171:30514/TCP 20s

View Pod Running Status

  1. # kubectl get pods
  2. NAME READY STATUS RESTARTS AGE
  3. mesha-b559fc4f4-m9752 2/2 Running 0 87s

Note: At this point, the service consumer has been deployed.

4. Service call verification

Access by domain name

  1. http://xxxx:30514/demo

The page prints the following error

Check the log and find that the domain name cannot be resolved:

  1. kubectl logs -f mesha-b559fc4f4-m9752 -n default  
  2.  
  3. 2021-11-05 09:01:37.372 WARN 1 --- [ault-executor-0] io.grpc.internal.ManagedChannelImpl : [Channel<1>: (dns:///AppMeshClient.mesh:50000)] Failed to resolve name. status=Status{code=UNAVAILABLE, description=Unable to resolve host AppMeshClient.mesh, cause=java.lang.RuntimeException: java.net.UnknownHostException: AppMeshClient.mesh: Name or service not known  
  4. at io.grpc.internal.DnsNameResolver.resolveAll(DnsNameResolver.java:436)
  5. at io.grpc.internal.DnsNameResolver$Resolve.resolveInternal(DnsNameResolver.java:272)
  6. at io.grpc.internal.DnsNameResolver$Resolve.run(DnsNameResolver.java:228)
  7. at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1149)
  8. at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:624)
  9. at java.lang.Thread.run(Thread.java:748)
  10. Caused by : java.net.UnknownHostException: AppMeshClient.mesh: Name   or service not known
  11. at java.net.Inet4AddressImpl.lookupAllHostAddr(Native Method)
  12. at java.net.InetAddress$2.lookupAllHostAddr(InetAddress.java:929)
  13. at java.net.InetAddress.getAddressesFromNameService(InetAddress.java:1324)
  14. at java.net.InetAddress.getAllByName0(InetAddress.java:1277)
  15. at java.net.InetAddress.getAllByName(InetAddress.java:1193)
  16. at java.net.InetAddress.getAllByName(InetAddress.java:1127)
  17. at io.grpc.internal.DnsNameResolver$JdkAddressResolver.resolveAddress(DnsNameResolver.java:646)
  18. at io.grpc.internal.DnsNameResolver.resolveAll(DnsNameResolver.java:404)
  19. ... 5 more
  20. }

Mapping the service provider IP through ServiceEntry

Check the IP address of the service provider meshb is xx0.17

  1. # kubectl get pods -o wide
  2. NAME READY STATUS RESTARTS AGE IP NODE NOMINATED NODE READINESS GATES
  3. mesha-b559fc4f4-m9752 2/2 Running 0 140m xx1.117 k8s-servicemesh-3 <none> <none>
  4. meshb-565945d794-wcb8z 2/2 Running 0 118m xx0.17 k8s-servicemesh-1 <none> <none>

meshb-service-entry.yaml content

  1. apiVersion: networking.istio.io/v1alpha3
  2. kind: ServiceEntry
  3. metadata:
  4. name : meshb
  5. spec:
  6. endpoints:
  7. - address: xx0.17
  8. hosts:
  9. - AppMeshClient.mesh
  10. location: MESH_INTERNAL
  11. ports:
  12. - name : grpc
  13. number: 50000
  14. protocol: grpc
  15. resolution: STATIC  

Execute the following command to make ServiceEntry effective

  1. kubectl apply -f meshb-service-entry.yaml
  2. serviceentry.networking.istio.io/meshb created

Visit the page again and find that it is normal

Note: At this point, the service consumer initiates a call to the service provider in the grid.

5. Log verification tracking

View the service consumer mesha log

  1. kubectl logs -f mesha-b559fc4f4-m9752 -n default  
  2.  
  3. I'm from server mesh demo!

Note: Information returned by the service consumer mesha from the service provider meshb.

View the service provider meshb log

  1. # kubectl logs -f meshb-565945d794-wcb8z -n default  
  2.  
  3. 2021-11-05 09:21:19.828 INFO 1 --- [main] osbwembedded.tomcat.TomcatWebServer: Tomcat started on port(s): 8181 (http) with context path ''  
  4. 2021-11-05 09:21:19.847 INFO 1 --- [main] com.melon.test.MeshbApplication : Started MeshbApplication in 3.859 seconds (JVM running for 4.344)  
  5. receiver client message: mesh demo!

Note: The service provider meshb received the request from the service consumer mesha.

View the envoy log of the service consumer mesha

  1. kubectl logs -f -l app=mesha -c istio-proxy -n default  
  2. 2021-11-05T10:25:17.772317Z info xdsproxy connected to upstream XDS server: istiod.istio-system.svc:15012
  3. 2021-11-05T10:52:35.856445Z warning envoy config StreamAggregatedResources gRPC config stream closed: 14, transport is closing
  4. 2021-11-05T10:52:36.093048Z info xdsproxy connected to upstream XDS server: istiod.istio-system.svc:15012
  5. [2021-11-05T11:06:23.510Z] "GET /demo HTTP/1.1" 500 - via_upstream - "-" 0 287 37 36 "-"   "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/95.0.4638.54 Safari/537.36"   "49cd74d8-86eb-4ef5-bd71-8236d188c2f9"   "10.69.31.156:30514"   "10.166.1.117:7171" inbound|7171|| 127.0.0.6:41007 10.166.1.117:7171 10.166.0.0:20826 - default  
  6. 2021-11-05T11:22:58.633956Z warning envoy config StreamAggregatedResources gRPC config stream closed: 14, transport is closing
  7. 2021-11-05T11:22:59.100387Z info xdsproxy connected to upstream XDS server: istiod.istio-system.svc:15012
  8. [2021-11-05T11:27:23.842Z] "POST /meshgrpc.Mesher/SayHello HTTP/2" 200 - via_upstream - "-" 17 33 371 319 "-"   "grpc-java-netty/1.28.0"   "79f2edbd-9c31-4265-87fb-38a594d9383b"   "AppMeshClient.mesh:50000"   "10.166.0.17:50000" outbound|50000||AppMeshClient.mesh 10.166.1.117:51914 240.240.0.63:50000 10.166.1.117:40544 - default  
  9. [2021-11-05T11:27:23.527Z] "GET /demo HTTP/1.1" 200 - via_upstream - "-" 0 26 729 728 "-"   "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/95.0.4638.54 Safari/537.36"   "859344b6-012d-4457-a391-2b4d13aa3e35"   "10.69.31.156:30514"   "10.166.1.117:7171" inbound|7171|| 127.0.0.6:47065 10.166.1.117:7171 10.166.0.0:2410 - default  

View the envoy log of the service consumer meshb

  1. # kubectl logs -f -l app=meshb -c istio-proxy -n default  
  2. 2021-11-05T09:57:12.490428Z warning envoy config StreamAggregatedResources gRPC config stream closed: 14, transport is closing
  3. 2021-11-05T09:57:12.765887Z info xdsproxy connected to upstream XDS server: istiod.istio-system.svc:15012
  4. 2021-11-05T10:24:48.767511Z warning envoy config StreamAggregatedResources gRPC config stream closed: 14, transport is closing
  5. 2021-11-05T10:24:48.914475Z info xdsproxy connected to upstream XDS server: istiod.istio-system.svc:15012
  6. 2021-11-05T10:57:52.604281Z warning envoy config StreamAggregatedResources gRPC config stream closed: 14, transport is closing
  7. 2021-11-05T10:57:52.757736Z info xdsproxy connected to upstream XDS server: istiod.istio-system.svc:15012
  8. 2021-11-05T11:26:56.551824Z warning envoy config StreamAggregatedResources gRPC config stream closed: 14, transport is closing
  9. 2021-11-05T11:26:56.987345Z info xdsproxy connected to upstream XDS server: istiod.istio-system.svc:15012
  10. [2021-11-05T11:27:23.844Z] "POST /meshgrpc.Mesher/SayHello HTTP/2" 200 - via_upstream - "-" 17 33 333 316 "-"   "grpc-java-netty/1.28.0"   "79f2edbd-9c31-4265-87fb-38a594d9383b"   "AppMeshClient.mesh:50000"   "10.166.0.17:50000" inbound|50000|| 127.0.0.6:44221 10.166.0.17:50000 10.166.1.117:51914 - default  

Log in to mesha's Pod to verify

  1. kubectl describe pod/mesha-b559fc4f4-m9752 -n default  
  2. [12:04:44root@mesha-b559fc4f4-m9752 /] C:2
  3. # curl -v AppMeshClient.mesh
  4. * About   connect () to AppMeshClient.mesh port 80 (#0)
  5. * Trying 240.240.0.63...
  6. * Connected to AppMeshClient.mesh (240.240.0.63) port 80 (#0)
  7. > GET / HTTP/1.1
  8. > User -Agent: curl/7.29.0
  9. > Host: AppMeshClient.mesh
  10. > Accept: */*
  11. >
  12. < HTTP/1.1 503 Service Unavailable
  13. < content-length: 91
  14. < content-type: text/plain
  15. < date : Fri, 05 Nov 2021 12:04:59 GMT
  16. < server:envoy
  17. <
  18. * Connection #0 to host AppMeshClient.mesh left intact
  19. upstream connect error or disconnect/reset before headers. reset reason: connection failure

Note: Log in to the Pod of mesha and access the domain name AppMeshClient.mesh. You will find that it automatically assigns IP "240.240.0.63" and points to the sidecar "envoy", which means that the traffic of the business container will be redirected to the sidecar through DNS Proxy.

Note: By checking the service consumer logs, service provider logs, and data plane enovy logs, it can be seen that the call is made in the istio mesh.

This article is reprinted from the WeChat public account "Guan Nong Lao Liang", which can be followed through the following QR code. To reprint this article, please contact the WeChat public account "Guan Nong Lao Liang".

<<:  Seven factors to consider in network redundancy design

>>:  HTML page basic structure and loading process

Recommend

In the 5G era, how do the three major operators fight the edge war?

5G makes the Internet of Everything possible. Whi...

WOT Huang Shuquan: Edge computing helps industrial intelligent manufacturing

[51CTO.com original article] On May 18-19, 2018, ...

5G network equipment security assessment escort "new infrastructure"

Unlike 4G mobile communication technology, which ...

Japan and Finland jointly develop 6G technology, Nokia will participate

Recently, foreign media reported that industry gr...

3G, 4G, 5G, why do we need so many "Gs"?

This can happen with our phones receiving signals...

Kerlink and Radio Bridge provide LoRaWAN solutions for private IoT networks

According to recent announcements, Kerlink and Ra...

From UML to SysML: The language journey of describing complex systems

In the vast world of systems engineering, which l...