199 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			199 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package grpc
 | |
| 
 | |
| // Copyright 2009 The Go Authors. All rights reserved.
 | |
| // Use of this source code is governed by a BSD-style
 | |
| // license that can be found in the LICENSE file.
 | |
| //
 | |
| // Meh, we need to get rid of this shit
 | |
| 
 | |
| import (
 | |
| 	"context"
 | |
| 	"errors"
 | |
| 	"reflect"
 | |
| 	"sync"
 | |
| 	"unicode"
 | |
| 	"unicode/utf8"
 | |
| 
 | |
| 	"github.com/micro/go-micro/v3/logger"
 | |
| 	"github.com/micro/go-micro/v3/server"
 | |
| )
 | |
| 
 | |
| var (
 | |
| 	// Precompute the reflect type for error. Can't use error directly
 | |
| 	// because Typeof takes an empty interface value. This is annoying.
 | |
| 	typeOfError = reflect.TypeOf((*error)(nil)).Elem()
 | |
| )
 | |
| 
 | |
| type methodType struct {
 | |
| 	method      reflect.Method
 | |
| 	ArgType     reflect.Type
 | |
| 	ReplyType   reflect.Type
 | |
| 	ContextType reflect.Type
 | |
| 	stream      bool
 | |
| }
 | |
| 
 | |
| type service struct {
 | |
| 	name   string                 // name of service
 | |
| 	rcvr   reflect.Value          // receiver of methods for the service
 | |
| 	typ    reflect.Type           // type of the receiver
 | |
| 	method map[string]*methodType // registered methods
 | |
| }
 | |
| 
 | |
| // server represents an RPC Server.
 | |
| type rServer struct {
 | |
| 	mu         sync.Mutex // protects the serviceMap
 | |
| 	serviceMap map[string]*service
 | |
| }
 | |
| 
 | |
| // Is this an exported - upper case - name?
 | |
| func isExported(name string) bool {
 | |
| 	rune, _ := utf8.DecodeRuneInString(name)
 | |
| 	return unicode.IsUpper(rune)
 | |
| }
 | |
| 
 | |
| // Is this type exported or a builtin?
 | |
| func isExportedOrBuiltinType(t reflect.Type) bool {
 | |
| 	for t.Kind() == reflect.Ptr {
 | |
| 		t = t.Elem()
 | |
| 	}
 | |
| 	// PkgPath will be non-empty even for an exported type,
 | |
| 	// so we need to check the type name as well.
 | |
| 	return isExported(t.Name()) || t.PkgPath() == ""
 | |
| }
 | |
| 
 | |
| // prepareEndpoint() returns a methodType for the provided method or nil
 | |
| // in case if the method was unsuitable.
 | |
| func prepareEndpoint(method reflect.Method) *methodType {
 | |
| 	mtype := method.Type
 | |
| 	mname := method.Name
 | |
| 	var replyType, argType, contextType reflect.Type
 | |
| 	var stream bool
 | |
| 
 | |
| 	// Endpoint() must be exported.
 | |
| 	if method.PkgPath != "" {
 | |
| 		return nil
 | |
| 	}
 | |
| 
 | |
| 	switch mtype.NumIn() {
 | |
| 	case 3:
 | |
| 		// assuming streaming
 | |
| 		argType = mtype.In(2)
 | |
| 		contextType = mtype.In(1)
 | |
| 		stream = true
 | |
| 	case 4:
 | |
| 		// method that takes a context
 | |
| 		argType = mtype.In(2)
 | |
| 		replyType = mtype.In(3)
 | |
| 		contextType = mtype.In(1)
 | |
| 	default:
 | |
| 		if logger.V(logger.ErrorLevel, logger.DefaultLogger) {
 | |
| 			logger.Errorf("method %v of %v has wrong number of ins: %v", mname, mtype, mtype.NumIn())
 | |
| 		}
 | |
| 		return nil
 | |
| 	}
 | |
| 
 | |
| 	if stream {
 | |
| 		// check stream type
 | |
| 		streamType := reflect.TypeOf((*server.Stream)(nil)).Elem()
 | |
| 		if !argType.Implements(streamType) {
 | |
| 			if logger.V(logger.ErrorLevel, logger.DefaultLogger) {
 | |
| 				logger.Errorf("%v argument does not implement Streamer interface: %v", mname, argType)
 | |
| 			}
 | |
| 			return nil
 | |
| 		}
 | |
| 	} else {
 | |
| 		// if not stream check the replyType
 | |
| 
 | |
| 		// First arg need not be a pointer.
 | |
| 		if !isExportedOrBuiltinType(argType) {
 | |
| 			if logger.V(logger.ErrorLevel, logger.DefaultLogger) {
 | |
| 				logger.Errorf("%v argument type not exported: %v", mname, argType)
 | |
| 			}
 | |
| 			return nil
 | |
| 		}
 | |
| 
 | |
| 		if replyType.Kind() != reflect.Ptr {
 | |
| 			if logger.V(logger.ErrorLevel, logger.DefaultLogger) {
 | |
| 				logger.Errorf("method %v reply type not a pointer: %v", mname, replyType)
 | |
| 			}
 | |
| 			return nil
 | |
| 		}
 | |
| 
 | |
| 		// Reply type must be exported.
 | |
| 		if !isExportedOrBuiltinType(replyType) {
 | |
| 			if logger.V(logger.ErrorLevel, logger.DefaultLogger) {
 | |
| 				logger.Errorf("method %v reply type not exported: %v", mname, replyType)
 | |
| 			}
 | |
| 			return nil
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	// Endpoint() needs one out.
 | |
| 	if mtype.NumOut() != 1 {
 | |
| 		if logger.V(logger.ErrorLevel, logger.DefaultLogger) {
 | |
| 			logger.Errorf("method %v has wrong number of outs: %v", mname, mtype.NumOut())
 | |
| 		}
 | |
| 		return nil
 | |
| 	}
 | |
| 	// The return type of the method must be error.
 | |
| 	if returnType := mtype.Out(0); returnType != typeOfError {
 | |
| 		if logger.V(logger.ErrorLevel, logger.DefaultLogger) {
 | |
| 			logger.Errorf("method %v returns %v not error", mname, returnType.String())
 | |
| 		}
 | |
| 		return nil
 | |
| 	}
 | |
| 	return &methodType{method: method, ArgType: argType, ReplyType: replyType, ContextType: contextType, stream: stream}
 | |
| }
 | |
| 
 | |
| func (server *rServer) register(rcvr interface{}) error {
 | |
| 	server.mu.Lock()
 | |
| 	defer server.mu.Unlock()
 | |
| 	if server.serviceMap == nil {
 | |
| 		server.serviceMap = make(map[string]*service)
 | |
| 	}
 | |
| 	s := new(service)
 | |
| 	s.typ = reflect.TypeOf(rcvr)
 | |
| 	s.rcvr = reflect.ValueOf(rcvr)
 | |
| 	sname := reflect.Indirect(s.rcvr).Type().Name()
 | |
| 	if sname == "" {
 | |
| 		logger.Fatalf("rpc: no service name for type %v", s.typ.String())
 | |
| 	}
 | |
| 	if !isExported(sname) {
 | |
| 		s := "rpc Register: type " + sname + " is not exported"
 | |
| 		if logger.V(logger.ErrorLevel, logger.DefaultLogger) {
 | |
| 			logger.Error(s)
 | |
| 		}
 | |
| 		return errors.New(s)
 | |
| 	}
 | |
| 	if _, present := server.serviceMap[sname]; present {
 | |
| 		return errors.New("rpc: service already defined: " + sname)
 | |
| 	}
 | |
| 	s.name = sname
 | |
| 	s.method = make(map[string]*methodType)
 | |
| 
 | |
| 	// Install the methods
 | |
| 	for m := 0; m < s.typ.NumMethod(); m++ {
 | |
| 		method := s.typ.Method(m)
 | |
| 		if mt := prepareEndpoint(method); mt != nil {
 | |
| 			s.method[method.Name] = mt
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if len(s.method) == 0 {
 | |
| 		s := "rpc Register: type " + sname + " has no exported methods of suitable type"
 | |
| 		if logger.V(logger.ErrorLevel, logger.DefaultLogger) {
 | |
| 			logger.Error(s)
 | |
| 		}
 | |
| 		return errors.New(s)
 | |
| 	}
 | |
| 	server.serviceMap[s.name] = s
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func (m *methodType) prepareContext(ctx context.Context) reflect.Value {
 | |
| 	if contextv := reflect.ValueOf(ctx); contextv.IsValid() {
 | |
| 		return contextv
 | |
| 	}
 | |
| 	return reflect.Zero(m.ContextType)
 | |
| }
 |