package server import ( "flag" pb "git.loafle.net/overflow/overflow_api_service/grpc" ws "git.loafle.net/overflow/overflow_api_service/ws" "github.com/golang/glog" "github.com/grpc-ecosystem/grpc-gateway/runtime" "golang.org/x/net/context" "google.golang.org/grpc" "net/http" "strings" "github.com/tmc/grpc-websocket-proxy/wsproxy" "log" "net" ) var ( overflowEndpoint = flag.String("echo_endpoint", ":9090", "/v1/overflow/services") grpcAddr = flag.String("grpcaddr", ":8001", "listen grpc addr") httpAddr = flag.String("addr", ":8000", "listen http addr") ) func RunGwRpc() (err error) { ctx := context.Background() ctx, cancel := context.WithCancel(ctx) defer cancel() if err := listenGRPC(*grpcAddr); err != nil { return err } mux := runtime.NewServeMux() opts := []grpc.DialOption{grpc.WithInsecure()} err = pb.RegisterOverflowGatewayHandlerFromEndpoint(ctx, mux, *overflowEndpoint, opts) if err != nil { return err } err = ws.RegisterWebSocketServiceHandlerFromEndpoint(ctx, mux, *grpcAddr, opts) if err != nil { return err } http.ListenAndServe(*httpAddr, wsproxy.WebsocketProxy(mux)) return http.ListenAndServe(":8080", wsproxy.WebsocketProxy(allowCORS(mux))) } func listenGRPC(listenAddr string) error { lis, err := net.Listen("tcp", listenAddr) if err != nil { return err } grpcServer := grpc.NewServer() ws.RegisterWebSocketServiceServer(grpcServer, &Server{}) go func() { if err := grpcServer.Serve(lis); err != nil { log.Println("serveGRPC err:", err) } }() return nil } //https://github.com/grpc-ecosystem/grpc-gateway/blob/master/examples/main.go func allowCORS(h http.Handler) http.Handler { return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { if origin := r.Header.Get("Origin"); origin != "" { w.Header().Set("Access-Control-Allow-Origin", origin) if r.Method == "OPTIONS" && r.Header.Get("Access-Control-Request-Method") != "" { preflightHandler(w, r) return } } h.ServeHTTP(w, r) }) } func preflightHandler(w http.ResponseWriter, r *http.Request) { headers := []string{"Content-Type", "Accept"} w.Header().Set("Access-Control-Allow-Headers", strings.Join(headers, ",")) methods := []string{"GET", "HEAD", "POST", "PUT", "DELETE"} w.Header().Set("Access-Control-Allow-Methods", strings.Join(methods, ",")) glog.Infof("preflight request for %s", r.URL.Path) return }