123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081 |
- package grpc // import "github.com/docker/docker/api/server/router/grpc"
- import (
- "context"
- "strings"
- "github.com/docker/docker/api/server/router"
- grpc_middleware "github.com/grpc-ecosystem/go-grpc-middleware"
- "github.com/moby/buildkit/util/grpcerrors"
- "github.com/moby/buildkit/util/tracing/detect"
- "github.com/sirupsen/logrus"
- "go.opentelemetry.io/contrib/instrumentation/google.golang.org/grpc/otelgrpc"
- "go.opentelemetry.io/otel/propagation"
- "go.opentelemetry.io/otel/trace"
- "golang.org/x/net/http2"
- "google.golang.org/grpc"
- )
- func init() {
- // enable in memory recording for grpc traces
- detect.Recorder = detect.NewTraceRecorder()
- }
- type grpcRouter struct {
- routes []router.Route
- grpcServer *grpc.Server
- h2Server *http2.Server
- }
- var propagators = propagation.NewCompositeTextMapPropagator(propagation.TraceContext{}, propagation.Baggage{})
- // NewRouter initializes a new grpc http router
- func NewRouter(backends ...Backend) router.Router {
- tp, err := detect.TracerProvider()
- if err != nil {
- logrus.WithError(err).Error("failed to detect trace provider")
- }
- opts := []grpc.ServerOption{grpc.UnaryInterceptor(grpcerrors.UnaryServerInterceptor), grpc.StreamInterceptor(grpcerrors.StreamServerInterceptor)}
- if tp != nil {
- streamTracer := otelgrpc.StreamServerInterceptor(otelgrpc.WithTracerProvider(tp), otelgrpc.WithPropagators(propagators))
- unary := grpc_middleware.ChainUnaryServer(unaryInterceptor(tp), grpcerrors.UnaryServerInterceptor)
- stream := grpc_middleware.ChainStreamServer(streamTracer, grpcerrors.StreamServerInterceptor)
- opts = []grpc.ServerOption{grpc.UnaryInterceptor(unary), grpc.StreamInterceptor(stream)}
- }
- r := &grpcRouter{
- h2Server: &http2.Server{},
- grpcServer: grpc.NewServer(opts...),
- }
- for _, b := range backends {
- b.RegisterGRPC(r.grpcServer)
- }
- r.initRoutes()
- return r
- }
- // Routes returns the available routers to the session controller
- func (gr *grpcRouter) Routes() []router.Route {
- return gr.routes
- }
- func (gr *grpcRouter) initRoutes() {
- gr.routes = []router.Route{
- router.NewPostRoute("/grpc", gr.serveGRPC),
- }
- }
- func unaryInterceptor(tp trace.TracerProvider) grpc.UnaryServerInterceptor {
- withTrace := otelgrpc.UnaryServerInterceptor(otelgrpc.WithTracerProvider(tp), otelgrpc.WithPropagators(propagators))
- return func(ctx context.Context, req interface{}, info *grpc.UnaryServerInfo, handler grpc.UnaryHandler) (resp interface{}, err error) {
- // This method is used by the clients to send their traces to buildkit so they can be included
- // in the daemon trace and stored in the build history record. This method can not be traced because
- // it would cause an infinite loop.
- if strings.HasSuffix(info.FullMethod, "opentelemetry.proto.collector.trace.v1.TraceService/Export") {
- return handler(ctx, req)
- }
- return withTrace(ctx, req, info, handler)
- }
- }
|