Skip to content

Commit d51071c

Browse files
trentmPeterEinberger
authored andcommitted
feat: Next.js server instrumentation (elastic#2959)
This adds instrumentation of the Next.js dev server (`next dev`) and prod server (`next start`). The APM transactions for incoming HTTP requests to the server will be named appropriately based on Next.js's routing -- both for user page routes (e.g. `GET /a-dynamic-page/[id]`) and for internal Next.js routes (e.g. `Next.js _next/data route my-page`, `Next.js Rewrite route /foo -> /bar`). As well, exceptions in server-side code (e.g. `getServerSideProps`, server-side run page handlers, API handlers) will be reported. This is a "technical preview", meaning we might change how this works in future versions -- including dropping the separate "start-next.js" start module. This also makes a change to stacktrace reporting. If a stack frame refers to a "*.min.js" file path and does not have source-map information, it is assumed to be a minimized file and source file context is *not* reported. This is to avoid excessively large stacktrace data. Reporting source file lines for name-minimized and 500+ character lines is not useful.
1 parent dc8f367 commit d51071c

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

53 files changed

+3186
-12
lines changed

.ci/.jenkins_tav.yml

+1
Original file line numberDiff line numberDiff line change
@@ -26,6 +26,7 @@ TAV:
2626
- mongodb-core
2727
- mysql
2828
- mysql2
29+
- next
2930
- pg
3031
- pug
3132
- redis

.eslintrc.json

+3
Original file line numberDiff line numberDiff line change
@@ -16,9 +16,12 @@
1616
"node_modules",
1717
"/examples/esbuild/dist",
1818
"/examples/typescript/dist",
19+
"/examples/nextjs",
1920
"/lib/opentelemetry-bridge/opentelemetry-core-mini",
2021
"/test/babel/out.js",
2122
"/test/lambda/fixtures/esbuild-bundled-handler/hello.js",
23+
"/test/instrumentation/modules/next/a-nextjs-app/pages",
24+
"/test/instrumentation/modules/next/a-nextjs-app/components",
2225
"/test/sourcemaps/fixtures/lib",
2326
"/test/sourcemaps/fixtures/src",
2427
"/test/stacktraces/fixtures/dist",

.github/dependabot.yml

+8
Original file line numberDiff line numberDiff line change
@@ -7,3 +7,11 @@ updates:
77
open-pull-requests-limit: 10
88
reviewers:
99
- "elastic/apm-agent-node-js"
10+
11+
- package-ecosystem: "npm"
12+
directory: "/test/instrumentation/modules/next/a-nextjs-app"
13+
schedule:
14+
interval: "weekly"
15+
open-pull-requests-limit: 5
16+
reviewers:
17+
- "elastic/apm-agent-node-js"

.gitignore

+1
Original file line numberDiff line numberDiff line change
@@ -14,3 +14,4 @@ node_modules
1414
/test/benchmarks/.tmp
1515
/tmp
1616
/examples/*/dist
17+
.next

CHANGELOG.asciidoc

+21
Original file line numberDiff line numberDiff line change
@@ -31,6 +31,7 @@ Notes:
3131
[[release-notes-3.x]]
3232
=== Node.js Agent version 3.x
3333
34+
3435
==== Unreleased
3536
3637
[float]
@@ -41,12 +42,31 @@ Notes:
4142
4243
* Enable support for redis v4 ({pull}2945[#2945])
4344
45+
* preview:[] Next.js server-side instrumentation. See the <<nextjs>> document.
46+
+
47+
This adds instrumentation of the Next.js dev server (`next dev`) and prod
48+
server (`next start`). The APM transactions for incoming HTTP requests to the
49+
server will be named appropriately based on Next.js's routing -- both for
50+
user page routes (e.g. `GET /a-dynamic-page/[id]`) and for internal Next.js
51+
routes (e.g. `Next.js _next/data route my-page`,
52+
`Next.js Rewrite route /foo -> /bar`). As well, exceptions in server-side code
53+
(e.g. `getServerSideProps`, server-side run page handlers, API handlers) will
54+
be reported. ({pull}2959[#2959])
55+
+
56+
This is a technical preview to get feedback from Next.js users. The details on
57+
how exactly the instrumentation works may change in future versions.
58+
4459
* Improve container-info gathering to support AWS ECS/Fargate environments.
4560
({issues}2914[#2914])
4661
4762
[float]
4863
===== Bug fixes
4964
65+
* Source lines of context in stacktraces is *no longer reported* for "*.min.js"
66+
files that do not have source-map information. These files are assumed to
67+
be minimized files, for which source line context won't be useful. This
68+
change is to guard against excessively large stacktrace data.
69+
5070
[float]
5171
===== Chores
5272
@@ -55,6 +75,7 @@ Notes:
5575
central config. The re-fetch delay is clamped to `[5 seconds, 1 day]`.
5676
({issues}2941[#2941])
5777
78+
5879
[[release-notes-3.39.0]]
5980
==== 3.39.0 2022/10/17
6081
126 KB
Loading

docs/nextjs.asciidoc

+208
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,208 @@
1+
:framework: Next.js
2+
3+
[[nextjs]]
4+
5+
ifdef::env-github[]
6+
NOTE: For the best reading experience,
7+
please view this documentation at https://www.elastic.co/guide/en/apm/agent/nodejs/current/nextjs.html[elastic.co]
8+
endif::[]
9+
10+
=== Get started with Next.js
11+
12+
The Elastic APM Node.js agent can be used to trace the Next.js server (`next
13+
start` or `next dev`) that runs your application without the need for code
14+
changes to your app. The APM transactions for incoming HTTP requests to the
15+
server will be named for the https://nextjs.org/docs/routing/introduction[pages]
16+
and https://nextjs.org/docs/api-routes/introduction[API endpoints] in your
17+
application, as well as for internal routes used by Next.js. Errors in code run
18+
on the server will be reported for viewing in the Kibana APM app.
19+
20+
Note that the Node.js APM agent can only instrument _server-side_ code. To
21+
monitor the client-side parts of a Next.js application, see the
22+
{apm-rum-ref}/intro.html[Elastic RUM agent].
23+
24+
NOTE: preview:[] This Next.js instrumentation is a _technical preview_ while we
25+
solicit feedback from Next.js users. If you are a Next.js user, please help us
26+
provide a better Next.js observability experience with your feedback on our
27+
https://discuss.elastic.co/tags/c/apm/nodejs[Discuss forum].
28+
29+
30+
[float]
31+
[[nextjs-prerequisites]]
32+
==== Prerequisites
33+
34+
You need an APM Server to send APM data to. Follow the
35+
{apm-guide-ref}/apm-quick-start.html[APM Quick start] if you have not set one up
36+
yet. You will need your *APM server URL* and an APM server *secret token* (or
37+
*API key*) for configuring the APM agent below.
38+
39+
You will also need a Next.js application to monitor. If you do not have an
40+
existing one to use, you can use the following to create a starter app (see
41+
https://nextjs.org/docs/getting-started[Next.js Getting Started docs] for more):
42+
43+
[source,bash]
44+
----
45+
npx create-next-app@latest # use the defaults
46+
cd my-app
47+
----
48+
49+
You can also take a look at and use this https://github.com/elastic/apm-agent-nodejs/tree/main/examples/nextjs/[Next.js + Elastic APM example app].
50+
51+
[float]
52+
[[nextjs-setup]]
53+
==== Step 1: Add the APM agent dependency
54+
55+
Add the `elastic-apm-node` module as a dependency to your application:
56+
57+
[source,bash]
58+
----
59+
npm install elastic-apm-node --save # or 'yarn add elastic-apm-node'
60+
----
61+
62+
63+
[float]
64+
==== Step 2: Start the APM agent
65+
66+
For the APM agent to instrument the Next.js server, it needs to be started
67+
before the Next.js server code is loaded. The best way to do so is by using
68+
Node's https://nodejs.org/api/cli.html#-r---require-module[`--require`] option
69+
to load the "elastic-apm-node/start-next.js" module -- this will start the agent
70+
(plus a little more for Next.js integration).
71+
72+
Edit the "dev" and "start" scripts in your "package.json" as follows:
73+
74+
[source,json]
75+
----
76+
{
77+
// ...
78+
"scripts": {
79+
"dev": "NODE_OPTIONS=--require=elastic-apm-node/start-next.js next dev",
80+
"build": "next build",
81+
"start": "NODE_OPTIONS=--require=elastic-apm-node/start-next.js next start",
82+
"lint": "next lint"
83+
},
84+
// ...
85+
}
86+
----
87+
88+
89+
[float]
90+
==== Step 3: Configure the APM agent
91+
92+
The APM agent can be
93+
<<configuring-the-agent,configured>>
94+
with environment variables or with an "elastic-apm-node.js" module in the
95+
current working directory. Note that because the APM agent is being loaded
96+
before the Next.js server, the
97+
https://nextjs.org/docs/basic-features/environment-variables[Next.js-supported
98+
".env" files] *cannot* be used to configure the APM agent. We will use an
99+
"elastic-apm-node.js" file here.
100+
101+
Create an "elastic-apm-node.js" file in the application root with the APM server
102+
URL and secret token values from the <<nextjs-prerequisites>> section above:
103+
104+
[source,javascript]
105+
----
106+
// elastic-apm-node.js
107+
module.exports = {
108+
serverUrl: 'https://...', // E.g. https://my-deployment-name.apm.us-west2.gcp.elastic-cloud.com
109+
secretToken: '...'
110+
}
111+
----
112+
113+
The equivalent using environment variables is:
114+
115+
[source,bash]
116+
----
117+
export ELASTIC_APM_SERVER_URL='https://...'
118+
export ELASTIC_APM_SECRET_TOKEN='...'
119+
----
120+
121+
See the <<configuration,agent configuration guide>> for full details on supported configuration variables.
122+
123+
124+
[float]
125+
==== Step 4: Start your Next.js app
126+
127+
[source,bash]
128+
----
129+
npm run dev # or 'npm run build && npm start' for the production server
130+
----
131+
132+
Open <http://localhost:3000> in your browser to load your Next.js app. If you
133+
used the `create-next-app` tool above, it defines an
134+
http://localhost:3000/api/hello[/api/hello] API endpoint. You can provide some
135+
artificial load by running the following in a separate terminal:
136+
137+
[source,bash]
138+
----
139+
while true; do sleep 1; curl -i http://localhost:3000/api/hello; done
140+
----
141+
142+
Visit your Kibana APM app and, after a few seconds, you should see a service
143+
entry for your Next.js app. The service name will be pulled from the "name"
144+
field in "package.json". It can be overriden with
145+
<<service-name,`serviceName`>>. Here is an example:
146+
147+
image::./images/nextjs-my-app-screenshot.png[Kibana APM app showing Next.js my-app]
148+
149+
150+
[float]
151+
[[nextjs-limitations]]
152+
==== Limitations and future work
153+
154+
This Next.js instrumentation has some limitations to be aware of.
155+
156+
Next.js build tooling bundles dependencies (using Webpack) for both client _and_
157+
server-side code execution. The Node.js APM agent does not work when bundled.
158+
See <<start-bundlers>> for details. The implication for Next.js instrumentation
159+
is that you cannot directly import and use the APM agent in your code. That
160+
means that using the <<agent-api>> for manual instrumentation is not currently
161+
possible.
162+
163+
This instrumentation supports naming APM transactions for many internal Next.js
164+
routes. For example, for
165+
https://nextjs.org/docs/basic-features/data-fetching/get-server-side-props[server-side
166+
rendering (SSR)] Next.js client code will make requests of the form `GET
167+
/next/_data/$buildId/$page.json`, for which the APM agent names the transaction
168+
`Next.js _next/data route $page`. However, there is a limitation with the
169+
Next.js "public folder catchall" route. HTTP requests that resolve to files in
170+
your "public/" directory, for example `GET /favicon.ico`, will result in a
171+
transaction named `GET unknown route`. See <<nextjs-unknown-routes>> below.
172+
173+
If you notice other limitations or have any suggestions, please give us feedback
174+
on our https://discuss.elastic.co/tags/c/apm/nodejs[Discuss forum].
175+
176+
177+
[float]
178+
[[nextjs-performance-monitoring]]
179+
==== Performance monitoring
180+
181+
Elastic APM automatically measures the performance of your Next.js application.
182+
It records spans for database queries, external HTTP requests, and other slow
183+
operations that happen during requests to your Next.js app. Spans are grouped in
184+
transactions -- by default one for each incoming HTTP request.
185+
186+
[float]
187+
[[nextjs-unknown-routes]]
188+
==== Unknown routes
189+
190+
include::./shared-set-up.asciidoc[tag=unknown-roots]
191+
192+
[float]
193+
[[nextjs-filter-sensitive-information]]
194+
==== Filter sensitive information
195+
196+
include::./shared-set-up.asciidoc[tag=filter-sensitive-info]
197+
198+
[float]
199+
[[nextjs-compatibility]]
200+
==== Compatibility
201+
202+
include::./shared-set-up.asciidoc[tag=compatibility-link]
203+
204+
[float]
205+
[[nextjs-troubleshooting]]
206+
==== Troubleshooting
207+
208+
include::./shared-set-up.asciidoc[tag=troubleshooting-link]

docs/set-up.asciidoc

+11-6
Original file line numberDiff line numberDiff line change
@@ -6,13 +6,15 @@ To get you off the ground, we've prepared guides for setting up the Agent with a
66
// This tagged region is used throughout the documentation to link to the framework guides
77
// Updates made here will be applied elsewhere as well.
88
// tag::web-frameworks-list[]
9+
* <<lambda>>
910
* <<express>>
11+
* <<fastify>>
1012
* <<hapi>>
1113
* <<koa>>
14+
* <<nextjs>>
1215
* <<restify>>
13-
* <<fastify>>
1416
* <<typescript>>
15-
* <<lambda>>
17+
* <<nextjs>>
1618
// end::web-frameworks-list[]
1719

1820
Alternatively, you can <<custom-stack>>.
@@ -26,21 +28,24 @@ Other useful documentation includes:
2628
* <<api>>
2729
* <<troubleshooting>>
2830

31+
include::./lambda.asciidoc[]
32+
2933
include::./express.asciidoc[]
3034

35+
include::./fastify.asciidoc[]
36+
3137
include::./hapi.asciidoc[]
3238

3339
include::./koa.asciidoc[]
3440

35-
include::./restify.asciidoc[]
41+
include::./nextjs.asciidoc[]
3642

37-
include::./fastify.asciidoc[]
43+
include::./restify.asciidoc[]
3844

3945
include::./typescript.asciidoc[]
4046

4147
include::./custom-stack.asciidoc[]
4248

43-
include::./lambda.asciidoc[]
4449

4550
[[starting-the-agent]]
4651
=== Starting the agent
@@ -97,7 +102,7 @@ A limitation of this approach is that you cannot configure the agent with an opt
97102
[[start-option-node-require-opt]]
98103
===== `node -r elastic-apm-node/start ...`
99104

100-
Another way to start the agent is with the `-r elastic-apm-node/start` https://nodejs.org/api/cli.html#-r---require-module[command line option to `node`]. This will import and start the APM agent before your application code starts. This method allows you to enable the agent _without touching any code_. This is the recommended start method for <<lambda,monitoring AWS Lambda functions>>.
105+
Another way to start the agent is with the `-r elastic-apm-node/start` https://nodejs.org/api/cli.html#-r---require-module[command line option to `node`]. This will import and start the APM agent before your application code starts. This method allows you to enable the agent _without touching any code_. This is the recommended start method for <<lambda,monitoring AWS Lambda functions>> and for tracing <<nextjs,a Next.js server>>.
101106

102107
[source,bash]
103108
----

docs/supported-technologies.asciidoc

+1-1
Original file line numberDiff line numberDiff line change
@@ -82,7 +82,6 @@ This agent is compatible with {apm-guide-ref}[APM Server] v6.6 and above.
8282

8383
Though you can use Elastic APM <<custom-stack,with any Node.js framework>>,
8484
we automate a few things for the most popular Node.js modules.
85-
8685
These are the frameworks that we officially support:
8786

8887
[options="header"]
@@ -94,6 +93,7 @@ These are the frameworks that we officially support:
9493
| <<hapi,@hapi/hapi>> | >=17.9.0 <21.0.0 |
9594
| <<hapi,hapi>> | >=9.0.0 <19.0.0 | Deprecated. No longer tested.
9695
| <<koa,Koa>> via koa-router or @koa/router | >=5.2.0 <13.0.0 | Koa doesn't have a built in router, so we can't support Koa directly since we rely on router information for full support. We currently support the most popular Koa router called https://github.com/koajs/koa-router[koa-router].
96+
| <<nextjs,Next.js>> | >=11.1.0 <14.0.0 | (Technical Preview) This instruments Next.js routing to name transactions for incoming HTTP transactions; and reports errors in user pages. It supports the Next.js production server (`next start`) and development server (`next dev`). See the <<nextjs,Getting Started document>>.
9797
| <<restify,Restify>> | >=5.2.0 |
9898
|=======================================================================
9999

examples/nextjs/.eslintrc.json

+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
{
2+
"extends": "next/core-web-vitals"
3+
}

0 commit comments

Comments
 (0)