mirror of
https://github.com/gusaul/grpcox.git
synced 2024-12-26 02:40:10 +00:00
232 lines
5.4 KiB
Go
232 lines
5.4 KiB
Go
package core
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"fmt"
|
|
"io"
|
|
"log"
|
|
"os"
|
|
"time"
|
|
|
|
"github.com/fullstorydev/grpcurl"
|
|
"github.com/jhump/protoreflect/desc"
|
|
"github.com/jhump/protoreflect/grpcreflect"
|
|
"google.golang.org/grpc"
|
|
"google.golang.org/grpc/codes"
|
|
"google.golang.org/grpc/metadata"
|
|
reflectpb "google.golang.org/grpc/reflection/grpc_reflection_v1alpha"
|
|
)
|
|
|
|
// Resource - hold 3 main function (List, Describe, and Invoke)
|
|
type Resource struct {
|
|
clientConn *grpc.ClientConn
|
|
descSource grpcurl.DescriptorSource
|
|
refClient *grpcreflect.Client
|
|
|
|
headers []string
|
|
md metadata.MD
|
|
}
|
|
|
|
//openDescriptor - use it to reflect server descriptor
|
|
func (r *Resource) openDescriptor() {
|
|
ctx := context.Background()
|
|
refCtx := metadata.NewOutgoingContext(ctx, r.md)
|
|
r.refClient = grpcreflect.NewClient(refCtx, reflectpb.NewServerReflectionClient(r.clientConn))
|
|
r.descSource = grpcurl.DescriptorSourceFromServer(ctx, r.refClient)
|
|
}
|
|
|
|
//closeDescriptor - please ensure to always close after open in the same flow
|
|
func (r *Resource) closeDescriptor() {
|
|
done := make(chan int)
|
|
go func() {
|
|
if r.refClient != nil {
|
|
r.refClient.Reset()
|
|
}
|
|
done <- 1
|
|
}()
|
|
|
|
select {
|
|
case <-done:
|
|
return
|
|
case <-time.After(3 * time.Second):
|
|
log.Printf("Reflection %s falied to close\n", r.clientConn.Target())
|
|
return
|
|
}
|
|
}
|
|
|
|
// List - To list all services exposed by a server
|
|
// symbol can be "" to list all available services
|
|
// symbol also can be service name to list all available method
|
|
func (r *Resource) List(symbol string) ([]string, error) {
|
|
r.openDescriptor()
|
|
defer r.closeDescriptor()
|
|
|
|
var result []string
|
|
if symbol == "" {
|
|
svcs, err := grpcurl.ListServices(r.descSource)
|
|
if err != nil {
|
|
return result, err
|
|
}
|
|
if len(svcs) == 0 {
|
|
return result, fmt.Errorf("No Services")
|
|
}
|
|
|
|
for _, svc := range svcs {
|
|
result = append(result, fmt.Sprintf("%s\n", svc))
|
|
}
|
|
} else {
|
|
methods, err := grpcurl.ListMethods(r.descSource, symbol)
|
|
if err != nil {
|
|
return result, err
|
|
}
|
|
if len(methods) == 0 {
|
|
return result, fmt.Errorf("No Function") // probably unlikely
|
|
}
|
|
|
|
for _, m := range methods {
|
|
result = append(result, fmt.Sprintf("%s\n", m))
|
|
}
|
|
}
|
|
|
|
return result, nil
|
|
}
|
|
|
|
// Describe - The "describe" verb will print the type of any symbol that the server knows about
|
|
// or that is found in a given protoset file.
|
|
// It also prints a description of that symbol, in the form of snippets of proto source.
|
|
// It won't necessarily be the original source that defined the element, but it will be equivalent.
|
|
func (r *Resource) Describe(symbol string) (string, string, error) {
|
|
r.openDescriptor()
|
|
defer r.closeDescriptor()
|
|
|
|
var result, template string
|
|
|
|
var symbols []string
|
|
if symbol != "" {
|
|
symbols = []string{symbol}
|
|
} else {
|
|
// if no symbol given, describe all exposed services
|
|
svcs, err := r.descSource.ListServices()
|
|
if err != nil {
|
|
return "", "", err
|
|
}
|
|
if len(svcs) == 0 {
|
|
log.Println("Server returned an empty list of exposed services")
|
|
}
|
|
symbols = svcs
|
|
}
|
|
for _, s := range symbols {
|
|
if s[0] == '.' {
|
|
s = s[1:]
|
|
}
|
|
|
|
dsc, err := r.descSource.FindSymbol(s)
|
|
if err != nil {
|
|
return "", "", err
|
|
}
|
|
|
|
txt, err := grpcurl.GetDescriptorText(dsc, r.descSource)
|
|
if err != nil {
|
|
return "", "", err
|
|
}
|
|
result = txt
|
|
|
|
if dsc, ok := dsc.(*desc.MessageDescriptor); ok {
|
|
// for messages, also show a template in JSON, to make it easier to
|
|
// create a request to invoke an RPC
|
|
tmpl := grpcurl.MakeTemplate(dsc)
|
|
_, formatter, err := grpcurl.RequestParserAndFormatterFor(grpcurl.Format("json"), r.descSource, true, false, nil)
|
|
if err != nil {
|
|
return "", "", err
|
|
}
|
|
str, err := formatter(tmpl)
|
|
if err != nil {
|
|
return "", "", err
|
|
}
|
|
template = str
|
|
}
|
|
}
|
|
|
|
return result, template, nil
|
|
}
|
|
|
|
// Invoke - invoking gRPC function
|
|
func (r *Resource) Invoke(ctx context.Context, symbol string, in io.Reader) (string, time.Duration, error) {
|
|
r.openDescriptor()
|
|
defer r.closeDescriptor()
|
|
|
|
// because of grpcurl directly fmt.Printf on their invoke function
|
|
// so we stub the Stdout using os.Pipe
|
|
backUpStdout := os.Stdout
|
|
defer func() {
|
|
os.Stdout = backUpStdout
|
|
}()
|
|
|
|
f, w, err := os.Pipe()
|
|
if err != nil {
|
|
return "", 0, err
|
|
}
|
|
os.Stdout = w
|
|
|
|
rf, formatter, err := grpcurl.RequestParserAndFormatterFor(grpcurl.Format("json"), r.descSource, false, true, in)
|
|
if err != nil {
|
|
return "", 0, err
|
|
}
|
|
h := grpcurl.NewDefaultEventHandler(os.Stdout, r.descSource, formatter, false)
|
|
|
|
start := time.Now()
|
|
err = grpcurl.InvokeRPC(ctx, r.descSource, r.clientConn, symbol, r.headers, h, rf.Next)
|
|
end := time.Now().Sub(start) / time.Millisecond
|
|
if err != nil {
|
|
return "", end, err
|
|
}
|
|
|
|
if h.Status.Code() != codes.OK {
|
|
return "", end, fmt.Errorf(h.Status.Message())
|
|
}
|
|
|
|
// copy the output in a separate goroutine so printing can't block indefinitely
|
|
outC := make(chan string)
|
|
go func() {
|
|
var buf bytes.Buffer
|
|
io.Copy(&buf, f)
|
|
outC <- buf.String()
|
|
}()
|
|
|
|
w.Close()
|
|
out := <-outC
|
|
|
|
return out, end, nil
|
|
}
|
|
|
|
// Close - to close all resources that was opened before
|
|
func (r *Resource) Close() {
|
|
done := make(chan int)
|
|
go func() {
|
|
if r.clientConn != nil {
|
|
r.clientConn.Close()
|
|
r.clientConn = nil
|
|
}
|
|
done <- 1
|
|
}()
|
|
|
|
select {
|
|
case <-done:
|
|
return
|
|
case <-time.After(3 * time.Second):
|
|
log.Printf("Connection %s falied to close\n", r.clientConn.Target())
|
|
return
|
|
}
|
|
}
|
|
|
|
func (r *Resource) isValid() bool {
|
|
return r.refClient != nil && r.clientConn != nil
|
|
}
|
|
|
|
func (r *Resource) exit(code int) {
|
|
// to force reset before os exit
|
|
r.Close()
|
|
os.Exit(code)
|
|
}
|