Skip to content
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

Add diagnostic channels #4429

Open
wants to merge 10 commits into
base: master
Choose a base branch
from

Conversation

YoannMa
Copy link

@YoannMa YoannMa commented Mar 2, 2023

Reimplementation of what @AdriVanHoudt made on the previous PR #4278

What changed since ?

  • Node 12 is no longer supported, there is no longer the issue of requiring a polyfill.
  • The API diagnostics_channel is stable since Node 16, The only part of the API we use, is identical in Node 14 but marked as experimental. Node 14 being EOL in less than 2 months, I don't really see a big issue there.

Following what Fastify does on their side https://github.com/fastify/fastify-diagnostics-channel, I added 3 more channels for the route, request and response.

I also added documentation of the exposed channels.

Questions :

  • Should we add more channels (onTimeout,onError) ?
  • Naming convention of the channel ?
    I copied Fastify naming scheme, but I don't really have any strong opinions regarding the naming scheme we pick : hapi:server, hapi.onServer or hapi.server

Cc'ing @Qard and @vdeturckheim as they are probably interested in the outcome of this PR.

API.md Outdated

## Diagnostic Channels

see : https://nodejs.org/docs/latest-v16.x/api/diagnostics_channel.html
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Instead of linking to the Node docs, can you provide a small example. If you still want to link to the Node docs, I would recommend linking to latest so we don't have to keep updating the link.

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Fixed the URL, added a bit of explaining and examples, do you feel like it's enough ?

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm good with it. Let's give other folks a chance to weigh in.

test/channels.js Outdated Show resolved Hide resolved
test/channels.js Outdated Show resolved Hide resolved
test/channels.js Show resolved Hide resolved
test/channels.js Outdated Show resolved Hide resolved
test/channels.js Outdated Show resolved Hide resolved
test/channels.js Outdated Show resolved Hide resolved
test/channels.js Outdated Show resolved Hide resolved
test/channels.js Outdated Show resolved Hide resolved
@YoannMa YoannMa marked this pull request as ready for review March 2, 2023 20:02
@Qard
Copy link

Qard commented Mar 3, 2023

It's been awhile since I've used Hapi so I'll have to take another look at it when I have time to know for sure if this satisfies what APMs need. One thing that does seem to be missing is we'd like an event to track errors.

@rochdev Any other suggestions for events that could help us?

@rochdev
Copy link

rochdev commented Mar 9, 2023

@Qard As you mentioned, definitely an event for errors. I would say also an event for when the route is entered as opposed to just created, unless the route is already available on the request object by the time it's emitted. Events for when plugins/extensions are entered/exited would also be useful, although this could be out of scope of this PR.

@YoannMa
Copy link
Author

YoannMa commented Mar 9, 2023

Route is available on the request object (see https://github.com/hapijs/hapi/blob/master/API.md#request.route) so I believe the hook on route entered is not necessary.

@Qard
Copy link

Qard commented Mar 9, 2023

I should also say, channel naming doesn't really matter as long as the chosen name is documented, and it includes the module name somehow to avoid possible collisions with other modules as diagnostics_channel is a global namespace.

It's also worth mentioning that I have an in-progress API to make the tracing use case easier with diagnostics_channel via nodejs/node#44943. While the API itself will not be available in all the supported versions of hapi for awhile yet, it does document the behaviour as a pattern which can be implemented manually with channels now, so long as you follow the guidelines of how it should behave. It could be worth a look to understand what things we capture and how that could be translated to channels introduced for hapi.

Copy link
Contributor

@kanongil kanongil left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Looks reasonable.

My main concern is that a throw in a registered channel handler can propagate to hapi. It seems that the current version guards against this, but a test to verify would be nice.

lib/core.js Outdated
@@ -510,6 +511,8 @@ exports = module.exports = internals.Core = class {

const request = Request.generate(this.root, req, res, options);

Channels.onRequestChannel.publish(request);
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I would probably move the request._execute() info assignment to the request constructor to make this more representative of what the onRequest ext sees.

Or maybe move the publish to the _execute() method? (though that would mean it doesn't see the ones that are denied due to load).

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

If this assignment is in the request._execute(), perhaps it is to avoid executing it in case of heavy load ?

I would vote to keep the publish() here given we need another event when a request is routed. I could add an event early in the request._lifecycle() method ? Something like hapi.onRequestLifecycle ?

lib/channels.js Outdated Show resolved Hide resolved
test/channels.js Outdated Show resolved Hide resolved
test/channels.js Outdated Show resolved Hide resolved
test/channels.js Outdated Show resolved Hide resolved
test/channels.js Outdated Show resolved Hide resolved
test/channels.js Outdated Show resolved Hide resolved
API.md Outdated Show resolved Hide resolved
API.md Show resolved Hide resolved
API.md Outdated
Comment on lines 5046 to 5051
const channel = DC.channel('hapi.onServer');

channel.subscribe((server) => {
// Do something with the server
console.log(server.version);
});
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hapi itself doesn't actually use any deprecated API, only these examples and the tests. I'm not a fan of using deprecated API to illustrate how to use, but since this is targeted at APM vendors they should known what they are doing anyway.

Maybe just have a single example at the top?

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I don't mind doing another PR when Hapi drops support for Node 14 to replace those deprecated usage

@kanongil
Copy link
Contributor

kanongil commented Mar 10, 2023

Route is available on the request object (see https://github.com/hapijs/hapi/blob/master/API.md#request.route) so I believe the hook on route entered is not necessary.

request.route won't be available yet when onRequest is triggered. The earliest it can be determined using public API, is with a onPreAuth hook, and it would have to contend with other hooks and not trigger when cookie processing fails.

@kanongil
Copy link
Contributor

kanongil commented Mar 10, 2023

Regarding channels, I would advocate on limiting it until there is a concrete need. Eg. with just the onServer channel, APM's should be able to hook something that matches hapi.onRoute, hapi.onRequest, and hapi.onResponse.

Exposing just a hapi.onServer channel would also mean that there is zero runtime overhead when channels are not used.

Hapi already contains most of the hooks a basic APM need, provided they get the Server object before the user can call methods on it. I don't see why we need to make it any easier for them.

@Qard
Copy link

Qard commented Mar 10, 2023

My main concern is that a throw in a registered channel handler can propagate to hapi. It seems that the current version guards against this, but a test to verify would be nice.

The diagnostics_channel API has guarded against this since the beginning. It's not possible for a throw in a subscriber to ever impact the publisher.

Exposing just a hapi.onServer channel would also mean that there is zero runtime overhead when channels are not used.

It's specifically designed to have zero overhead when a channel is not used. The publish function is a no-op until the channel has subscribers, and if it ever returns to zero subscribers it becomes a no-op again.

Copy link
Contributor

@kanongil kanongil left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Looking at it again I think it would work better if the channels are triggered before the equivalent user-facing hook. This way the event handlers cannot modify the object before the APM sees it, making it a more "pure" signal.

Test to verify this would also be nice.

@YoannMa
Copy link
Author

YoannMa commented Mar 19, 2023

Added both hapi.onRequestLifecycle and hapi.onError channels.

I'm not really sure about the name hapi.onRequestLifecycle, if you have some ideas for another name, I'm listening.

@kanongil
Copy link
Contributor

kanongil commented Mar 21, 2023

I would much prefer that this PR does not add any more channel hooks. Ideally each hook gets a PR to allow simpler reviews. Eg. i have comments regarding hapi.onError, but I don't want to complicate the review further.

Personally, I would just include the bare minimum (ie. hapi.onServer) for an initial PR. As stated in #4429 (comment), it should be enough to get started. For instance, looking at the elastic-apm-node APM it is the only hook it will need. The remaining hooks are handled using the public API.

@Qard
Copy link

Qard commented Mar 21, 2023

There's typically greater overhead to needing to hook into some public API like that though, and a greater risk of it changing between versions. APMs would generally prefer that data be published directly. While having only the hapi.onServer event is better than nothing, it's not actually what we want and we wouldn't usually even need the hapi.onServer event if we have the other events.

@kanongil kanongil added the feature New functionality or improvement label Nov 6, 2023
@kanongil kanongil changed the title feat: add diagnostic channel for server, route, request creation and response sent Add diagnostic channels Nov 6, 2023
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
feature New functionality or improvement
Projects
None yet
Development

Successfully merging this pull request may close these issues.

None yet

5 participants