-
Notifications
You must be signed in to change notification settings - Fork 377
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Support gRPC #379
Comments
@delner I'm going to pull the higher-level feedback and considerations into this issue. I have many of the changes you've requested implemented, though not yet pushed up. I'm thinking of opening a new PR with a simpler, more complete integration and closing #378. My intention is to get all the items below completed by end of this week, early next week at the latest. I'll be testing some of our newer internal services at KennaSecurity on my branch as well as providing a simple demo application for inspection. TODO & FollowupInterception Patterns:
Testing:
Datadog Features:
|
@Jared-Prime Great! Thanks for putting this together. Looking forward to getting this wrapped up. |
demo app + performance benchmark: https://github.com/Jared-Prime/grpc-demo |
I feel like #403 is in a good place. I think I've gotten the |
@Jared-Prime It's looking pretty good, but I don't think the use of You have this to set the pin: def add_pin
Pin.new(
get_option(:service_name),
app: 'grpc',
app_type: 'grpc',
tracer: get_option(:tracer)
).onto(::GRPC)
end And this to get the pin back: def pin
Pin.get_from(::GRPC)
end This effectively globalizes the configuration settings by attaching them to that constant, meaning if you do gRPC calls to two different services, you'd end up seeing them traced as one service (which is not ideal.) I suspect you probably were using an older integration in our library for inspiration? The pin used to be the primary means of apply configuration prior to our configuration and The tricky part here is putting the pin on the right thing... what the "right" thing is might depend on how this gRPC library works. Usually the "right" thing would be something unique to a service, e.g. a client object configured with specific server settings. Such that in the intercept function, you could use some property of the request to retrieve the configuration specific for that kind of request. I don't know if there's anything like that here; it looks like a middleware pattern, that may or may not expose a "client" object or some other suitable attribute of the request. |
This is tricky. In gRPC Ruby, the developer retrieves client classes using the The gRPC Go package for DataDog supplying the intereceptor when setting up the client. // from the godocs at https://godoc.org/github.com/DataDog/dd-trace-go/contrib/google.golang.org/grpc
// Create the client interceptor using the grpc trace package.
i := grpctrace.UnaryClientInterceptor(
grpctrace.WithServiceName("my-grpc-client"),
grpctrace.WithTracer(tracer.DefaultTracer),
)
// Create initialization options for dialing into a server. Make sure
// to include the created interceptor.
opts := []grpc.DialOption{
grpc.WithInsecure(),
grpc.WithUnaryInterceptor(i),
}
// Dial in...
conn, err := grpc.Dial("localhost:50051", opts...)
if err != nil {
log.Fatal(err)
}
defer conn.Close()
// And continue using the connection as normal. In Ruby, that would look like: Demo::Echo::Service.rpc_stub_class
.new('localhost:50051',
:this_channel_is_insecure,
:interceptors => [Datadog::Contrib::GRPC::DatadogInterceptor::Client.new]) Pin information may then be set on the interceptor instance. What do you think of that approach? Would that also mean there's no need for a global pin or monkey patching the |
Ah, if you can initialize and assign a separate interceptor instance per service, that could work well. The only question left would be how are users expected to configure this behavior? If at all possible we'd like to provide a default Could you provide example code for 1) basic, single service setup and 2) multi-service setup? As in, what it would look like for a user to simply use the gRPC tracing function we're building here? |
Here's what I came up with in 9573b5d Datadog.configure do |c|
c.use :grpc, service_name: "Default"
end
default_client = Demo::Echo::Service.rpc_stub_class.new(
'localhost:50051',
:this_channel_is_insecure
)
configured_interceptor = Datadog::Contrib::GRPC::DatadogInterceptor::Client.new do |c|
c.service_name = "Alternate"
end
alternate_client = Demo::Echo::Service.rpc_stub_class.new(
'localhost:50051',
:this_channel_is_insecure,
:interceptors => [configured_interceptor]
) Spans sent by |
@Jared-Prime Awesome, I think that looks good! It should allow for the multiple service scenario. |
This has been merged to |
As a user of gRPC services/clients and of Datadog APM, I would like to have the gem support a grpc plugin.
Proposed Implementation: #378
The text was updated successfully, but these errors were encountered: