|
| 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] |
0 commit comments