examples: add client/server to be used for xds examples (#3362)
This commit is contained in:
37
examples/features/xds/README.md
Normal file
37
examples/features/xds/README.md
Normal file
@ -0,0 +1,37 @@
|
|||||||
|
# gRPC xDS example
|
||||||
|
|
||||||
|
xDS is the protocol initially used by Envoy, that is evolving into a universal
|
||||||
|
data plan API for service mesh.
|
||||||
|
|
||||||
|
The xDS example is a Hello World client/server capable of being configured with
|
||||||
|
the XDS management protocol. Out-of-the-box it behaves the same as [our other
|
||||||
|
hello world
|
||||||
|
example](https://github.com/grpc/grpc-go/tree/master/examples/helloworld). The
|
||||||
|
server replies with responses including its hostname.
|
||||||
|
|
||||||
|
**Note** that xDS support is incomplete and experimental, with limited
|
||||||
|
compatibility.
|
||||||
|
|
||||||
|
## xDS environment setup
|
||||||
|
|
||||||
|
This example doesn't include instuctions to setup xDS environment. Please
|
||||||
|
refer to documentation specific for your xDS management server.
|
||||||
|
|
||||||
|
The client also needs a bootstrap file. See [gRFC
|
||||||
|
A27](https://github.com/grpc/proposal/pull/170/files#diff-05ea4a5894abbc0261b006741220598cR100)
|
||||||
|
for the bootstrap format.
|
||||||
|
|
||||||
|
## The client
|
||||||
|
|
||||||
|
The client application needs to import the xDS package to install the resolver and balancers:
|
||||||
|
|
||||||
|
```go
|
||||||
|
_ "google.golang.org/grpc/xds/experimental" // To install the xds resolvers and balancers.
|
||||||
|
```
|
||||||
|
|
||||||
|
Then, use `xds-experimental` target scheme for the ClientConn.
|
||||||
|
|
||||||
|
```
|
||||||
|
$ export GRPC_XDS_BOOTSTRAP=/path/to/bootstrap.json
|
||||||
|
$ go run client/main.go "xDS world" xds-experimental:///target_service
|
||||||
|
```
|
97
examples/features/xds/client/main.go
Normal file
97
examples/features/xds/client/main.go
Normal file
@ -0,0 +1,97 @@
|
|||||||
|
// +build go1.11
|
||||||
|
|
||||||
|
/*
|
||||||
|
*
|
||||||
|
* Copyright 2020 gRPC authors.
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
// Package main implements a client for Greeter service.
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"flag"
|
||||||
|
"fmt"
|
||||||
|
"log"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"google.golang.org/grpc"
|
||||||
|
pb "google.golang.org/grpc/examples/helloworld/helloworld"
|
||||||
|
|
||||||
|
_ "google.golang.org/grpc/xds/experimental" // To install the xds resolvers and balancers.
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
defaultTarget = "localhost:50051"
|
||||||
|
defaultName = "world"
|
||||||
|
)
|
||||||
|
|
||||||
|
var help = flag.Bool("help", false, "Print usage information")
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
flag.Usage = func() {
|
||||||
|
fmt.Fprintf(flag.CommandLine.Output(), `
|
||||||
|
Usage: client [name [target]]
|
||||||
|
|
||||||
|
name
|
||||||
|
The name you wish to be greeted by. Defaults to %q
|
||||||
|
target
|
||||||
|
The URI of the server, e.g. "xds-experimental:///helloworld-service". Defaults to %q
|
||||||
|
`, defaultName, defaultTarget)
|
||||||
|
|
||||||
|
flag.PrintDefaults()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
flag.Parse()
|
||||||
|
if *help {
|
||||||
|
flag.Usage()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
args := flag.Args()
|
||||||
|
|
||||||
|
if len(args) > 2 {
|
||||||
|
flag.Usage()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
name := defaultName
|
||||||
|
if len(args) > 0 {
|
||||||
|
name = args[0]
|
||||||
|
}
|
||||||
|
|
||||||
|
target := defaultTarget
|
||||||
|
if len(args) > 1 {
|
||||||
|
target = args[1]
|
||||||
|
}
|
||||||
|
|
||||||
|
// Set up a connection to the server.
|
||||||
|
conn, err := grpc.Dial(target, grpc.WithInsecure())
|
||||||
|
if err != nil {
|
||||||
|
log.Fatalf("did not connect: %v", err)
|
||||||
|
}
|
||||||
|
defer conn.Close()
|
||||||
|
c := pb.NewGreeterClient(conn)
|
||||||
|
|
||||||
|
ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second)
|
||||||
|
defer cancel()
|
||||||
|
r, err := c.SayHello(ctx, &pb.HelloRequest{Name: name})
|
||||||
|
if err != nil {
|
||||||
|
log.Fatalf("could not greet: %v", err)
|
||||||
|
}
|
||||||
|
log.Printf("Greeting: %s", r.GetMessage())
|
||||||
|
}
|
129
examples/features/xds/server/main.go
Normal file
129
examples/features/xds/server/main.go
Normal file
@ -0,0 +1,129 @@
|
|||||||
|
// +build go1.11
|
||||||
|
|
||||||
|
/*
|
||||||
|
*
|
||||||
|
* Copyright 2020 gRPC authors.
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
// Package main starts Greeter service that will response with the hostname.
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"flag"
|
||||||
|
"fmt"
|
||||||
|
"log"
|
||||||
|
"math/rand"
|
||||||
|
"net"
|
||||||
|
"os"
|
||||||
|
"strconv"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"google.golang.org/grpc"
|
||||||
|
pb "google.golang.org/grpc/examples/helloworld/helloworld"
|
||||||
|
)
|
||||||
|
|
||||||
|
var help = flag.Bool("help", false, "Print usage information")
|
||||||
|
|
||||||
|
const (
|
||||||
|
defaultPort = 50051
|
||||||
|
)
|
||||||
|
|
||||||
|
// server is used to implement helloworld.GreeterServer.
|
||||||
|
type server struct {
|
||||||
|
pb.UnimplementedGreeterServer
|
||||||
|
|
||||||
|
serverName string
|
||||||
|
}
|
||||||
|
|
||||||
|
func newServer(serverName string) *server {
|
||||||
|
return &server{
|
||||||
|
serverName: serverName,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// SayHello implements helloworld.GreeterServer
|
||||||
|
func (s *server) SayHello(ctx context.Context, in *pb.HelloRequest) (*pb.HelloReply, error) {
|
||||||
|
log.Printf("Received: %v", in.GetName())
|
||||||
|
return &pb.HelloReply{Message: "Hello " + in.GetName() + ", from " + s.serverName}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func determineHostname() string {
|
||||||
|
hostname, err := os.Hostname()
|
||||||
|
if err != nil {
|
||||||
|
log.Printf("Failed to get hostname: %v, will generate one", err)
|
||||||
|
rand.Seed(time.Now().UnixNano())
|
||||||
|
return fmt.Sprintf("generated-%03d", rand.Int()%100)
|
||||||
|
}
|
||||||
|
return hostname
|
||||||
|
}
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
flag.Usage = func() {
|
||||||
|
fmt.Fprintf(flag.CommandLine.Output(), `
|
||||||
|
Usage: server [port [hostname]]
|
||||||
|
|
||||||
|
port
|
||||||
|
The listen port. Defaults to %d
|
||||||
|
hostname
|
||||||
|
The name clients will see in greet responses. Defaults to the machine's hostname
|
||||||
|
`, defaultPort)
|
||||||
|
|
||||||
|
flag.PrintDefaults()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
flag.Parse()
|
||||||
|
if *help {
|
||||||
|
flag.Usage()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
args := flag.Args()
|
||||||
|
|
||||||
|
if len(args) > 2 {
|
||||||
|
flag.Usage()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
port := defaultPort
|
||||||
|
if len(args) > 0 {
|
||||||
|
var err error
|
||||||
|
port, err = strconv.Atoi(args[0])
|
||||||
|
if err != nil {
|
||||||
|
log.Printf("Invalid port number: %v", err)
|
||||||
|
flag.Usage()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var hostname string
|
||||||
|
if len(args) > 1 {
|
||||||
|
hostname = args[1]
|
||||||
|
}
|
||||||
|
if hostname == "" {
|
||||||
|
hostname = determineHostname()
|
||||||
|
}
|
||||||
|
|
||||||
|
lis, err := net.Listen("tcp", fmt.Sprintf("localhost:%d", port))
|
||||||
|
if err != nil {
|
||||||
|
log.Fatalf("failed to listen: %v", err)
|
||||||
|
}
|
||||||
|
s := grpc.NewServer()
|
||||||
|
pb.RegisterGreeterServer(s, newServer(hostname))
|
||||||
|
log.Printf("serving on %s, hostname %s", lis.Addr(), hostname)
|
||||||
|
s.Serve(lis)
|
||||||
|
}
|
Reference in New Issue
Block a user