| 
									
										
										
										
											2025-03-29 15:02:49 +08:00
										 |  |  | package service | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							| 
									
										
										
										
											2025-03-30 13:23:46 +08:00
										 |  |  | 	"context" | 
					
						
							| 
									
										
										
										
											2025-03-29 15:02:49 +08:00
										 |  |  | 	"log" | 
					
						
							|  |  |  | 	"net" | 
					
						
							|  |  |  | 	"os" | 
					
						
							|  |  |  | 	"strconv" | 
					
						
							|  |  |  | 	"strings" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-30 13:23:46 +08:00
										 |  |  | 	"net/http" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	gwRuntime "github.com/grpc-ecosystem/grpc-gateway/v2/runtime" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-29 15:02:49 +08:00
										 |  |  | 	"git.apinb.com/bsm-sdk/core/conf" | 
					
						
							|  |  |  | 	"git.apinb.com/bsm-sdk/core/env" | 
					
						
							| 
									
										
										
										
											2025-09-27 00:20:36 +08:00
										 |  |  | 	"git.apinb.com/bsm-sdk/core/printer" | 
					
						
							| 
									
										
										
										
											2025-03-29 15:02:49 +08:00
										 |  |  | 	"git.apinb.com/bsm-sdk/core/vars" | 
					
						
							|  |  |  | 	clientv3 "go.etcd.io/etcd/client/v3" | 
					
						
							|  |  |  | 	"google.golang.org/grpc" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type ( | 
					
						
							|  |  |  | 	// RegisterFn defines the method to register a server.
 | 
					
						
							|  |  |  | 	RegisterFn func(*grpc.Server) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	Service struct { | 
					
						
							|  |  |  | 		GrpcSrv *grpc.Server | 
					
						
							|  |  |  | 		Opts    *Options | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	Options struct { | 
					
						
							| 
									
										
										
										
											2025-03-30 13:23:46 +08:00
										 |  |  | 		Addr        string | 
					
						
							|  |  |  | 		EtcdClient  *clientv3.Client | 
					
						
							|  |  |  | 		MsConf      *conf.MicroServiceConf | 
					
						
							|  |  |  | 		GatewayConf *conf.GatewayConf | 
					
						
							|  |  |  | 		GatewayCtx  context.Context | 
					
						
							|  |  |  | 		GatewayMux  *gwRuntime.ServeMux | 
					
						
							| 
									
										
										
										
											2025-03-29 15:02:49 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func New(srv *grpc.Server, opts *Options) *Service { | 
					
						
							|  |  |  | 	return &Service{GrpcSrv: srv, Opts: opts} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-27 00:41:27 +08:00
										 |  |  | // Addr formats ip and port into host:port.
 | 
					
						
							| 
									
										
										
										
											2025-03-29 15:02:49 +08:00
										 |  |  | func Addr(ip string, port int) string { | 
					
						
							|  |  |  | 	return net.JoinHostPort(ip, strconv.Itoa(port)) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (s *Service) Start() { | 
					
						
							| 
									
										
										
										
											2025-09-27 00:20:36 +08:00
										 |  |  | 	printer.Info("[BSM - %s] Service Starting ...", vars.ServiceKey) | 
					
						
							| 
									
										
										
										
											2025-03-29 15:02:49 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// register to etcd.
 | 
					
						
							| 
									
										
										
										
											2025-03-30 13:23:46 +08:00
										 |  |  | 	if s.Opts.MsConf != nil && s.Opts.MsConf.Enable { | 
					
						
							| 
									
										
										
										
											2025-03-29 15:02:49 +08:00
										 |  |  | 		if s.Opts.EtcdClient == nil { | 
					
						
							| 
									
										
										
										
											2025-09-27 00:20:36 +08:00
										 |  |  | 			printer.Error("[BSM Register] Etcd Client is nil.") | 
					
						
							| 
									
										
										
										
											2025-03-29 15:02:49 +08:00
										 |  |  | 			os.Exit(1) | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2025-09-27 00:20:36 +08:00
										 |  |  | 		printer.Info("[BSM - %s] Registering Service to Etcd ...", vars.ServiceKey) | 
					
						
							| 
									
										
										
										
											2025-09-27 00:41:27 +08:00
										 |  |  | 		// get methods for gateway/router discovery
 | 
					
						
							| 
									
										
										
										
											2025-03-29 15:02:49 +08:00
										 |  |  | 		methods := FoundGrpcMethods(s.GrpcSrv) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// set router key
 | 
					
						
							|  |  |  | 		routerKey := vars.ServiceRootPrefix + "Router/" + env.Runtime.Workspace + "/" + strings.ToLower(vars.ServiceKey) + "/" + s.Opts.Addr | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-27 00:41:27 +08:00
										 |  |  | 		// register to etcd with lease
 | 
					
						
							| 
									
										
										
										
											2025-03-29 15:02:49 +08:00
										 |  |  | 		register, err := RegisterService(s.Opts.EtcdClient, routerKey, methods, vars.ServiceLease) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			log.Panicf("[ERROR] %s Service Register:%s \n", vars.ServiceKey, err.Error()) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		anonKey := vars.ServiceRootPrefix + "Router/" + env.Runtime.Workspace + "/" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-30 13:23:46 +08:00
										 |  |  | 		register.SetAnonymous(anonKey, s.Opts.MsConf.Anonymous) | 
					
						
							| 
									
										
										
										
											2025-03-29 15:02:49 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		// service register lease
 | 
					
						
							|  |  |  | 		go register.ListenLeaseRespChan() | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// run grpc srv.
 | 
					
						
							|  |  |  | 	tcpListen, err := net.Listen("tcp", s.Opts.Addr) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		panic(err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-30 13:23:46 +08:00
										 |  |  | 	go func() { | 
					
						
							|  |  |  | 		if err := s.GrpcSrv.Serve(tcpListen); err != nil { | 
					
						
							|  |  |  | 			panic(err) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	}() | 
					
						
							| 
									
										
										
										
											2025-09-27 00:20:36 +08:00
										 |  |  | 	printer.Success("[BSM - %s] Grpc %s Runing Success !", vars.ServiceKey, s.Opts.Addr) | 
					
						
							| 
									
										
										
										
											2025-03-30 13:23:46 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if s.Opts.GatewayConf != nil && s.Opts.GatewayConf.Enable { | 
					
						
							|  |  |  | 		addr := Addr("0.0.0.0", s.Opts.GatewayConf.Port) | 
					
						
							|  |  |  | 		go s.Gateway(s.Opts.Addr, addr) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-27 00:20:36 +08:00
										 |  |  | 		printer.Success("[BSM - %s] Http %s Runing Success!", vars.ServiceKey, addr) | 
					
						
							| 
									
										
										
										
											2025-03-29 15:02:49 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-30 13:23:46 +08:00
										 |  |  | 	select {} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (s *Service) Gateway(grpcAddr string, httpAddr string) { | 
					
						
							|  |  |  | 	// 1. 定义一个context
 | 
					
						
							|  |  |  | 	_, cancel := context.WithCancel(s.Opts.GatewayCtx) | 
					
						
							|  |  |  | 	defer cancel() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-27 00:41:27 +08:00
										 |  |  | 	// 不因 HTTP 启动失败而导致 panic
 | 
					
						
							|  |  |  | 	if err := http.ListenAndServe(httpAddr, s.Opts.GatewayMux); err != nil { | 
					
						
							|  |  |  | 		printer.Error("[BSM - %s] Http Serve Error: %v", vars.ServiceKey, err) | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2025-03-29 15:02:49 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-23 12:41:31 +08:00
										 |  |  | func (s *Service) Use(initFunc func() error) { | 
					
						
							|  |  |  | 	err := (initFunc)() | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2025-09-27 00:20:36 +08:00
										 |  |  | 		printer.Error(err.Error()) | 
					
						
							| 
									
										
										
										
											2025-09-23 12:41:31 +08:00
										 |  |  | 		panic(err) | 
					
						
							| 
									
										
										
										
											2025-09-23 12:37:00 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-29 15:02:49 +08:00
										 |  |  | func (s *Service) Stop() { | 
					
						
							|  |  |  | 	s.GrpcSrv.GracefulStop() | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // found grpc methods.
 | 
					
						
							|  |  |  | func FoundGrpcMethods(s *grpc.Server) []string { | 
					
						
							|  |  |  | 	var mothods []string | 
					
						
							|  |  |  | 	for key, srv := range s.GetServiceInfo() { | 
					
						
							|  |  |  | 		srvName := strings.Split(key, ".")[1] | 
					
						
							|  |  |  | 		for _, mn := range srv.Methods { | 
					
						
							|  |  |  | 			mothods = append(mothods, srvName+"."+mn.Name) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return mothods | 
					
						
							|  |  |  | } |