---
title: Custom Server
description: Start a Next.js app programmatically using a custom server.
url: "https://nextjs.org/docs/14/pages/building-your-application/configuring/custom-server"
version: 14.2.35
lastUpdated: 2024-03-22
prerequisites:
  - "Building Your Application: /docs/14/pages/building-your-application"
  - "Configuring: /docs/14/pages/building-your-application/configuring"
---


<details>
<summary>Examples</summary>

* [Custom Server](https://github.com/vercel/next.js/tree/canary/examples/custom-server)
* [SSR Caching](https://github.com/vercel/next.js/tree/canary/examples/ssr-caching)

</details>

By default, Next.js includes its own server with `next start`. If you have an existing backend, you can still use it with Next.js (this is not a custom server). A custom Next.js server allows you to start a server 100% programmatically in order to use custom server patterns. Most of the time, you will not need this - but it's available for complete customization.

> **Good to know**:
>
> * Before deciding to use a custom server, please keep in mind that it should only be used when the integrated router of Next.js can't meet your app requirements. A custom server will remove important performance optimizations, like **serverless functions** and **[Automatic Static Optimization](/docs/pages/building-your-application/rendering/automatic-static-optimization).**
> * A custom server **cannot** be deployed on [Vercel](https://vercel.com/solutions/nextjs).

Take a look at the following example of a custom server:

```js filename="server.js"
const { createServer } = require('http')
const { parse } = require('url')
const next = require('next')

const dev = process.env.NODE_ENV !== 'production'
const hostname = 'localhost'
const port = 3000
// when using middleware `hostname` and `port` must be provided below
const app = next({ dev, hostname, port })
const handle = app.getRequestHandler()

app.prepare().then(() => {
  createServer(async (req, res) => {
    try {
      // Be sure to pass `true` as the second argument to `url.parse`.
      // This tells it to parse the query portion of the URL.
      const parsedUrl = parse(req.url, true)
      const { pathname, query } = parsedUrl

      if (pathname === '/a') {
        await app.render(req, res, '/a', query)
      } else if (pathname === '/b') {
        await app.render(req, res, '/b', query)
      } else {
        await handle(req, res, parsedUrl)
      }
    } catch (err) {
      console.error('Error occurred handling', req.url, err)
      res.statusCode = 500
      res.end('internal server error')
    }
  })
    .once('error', (err) => {
      console.error(err)
      process.exit(1)
    })
    .listen(port, () => {
      console.log(`> Ready on http://${hostname}:${port}`)
    })
})
```

> `server.js` doesn't go through babel or webpack. Make sure the syntax and sources this file requires are compatible with the current node version you are running.

To run the custom server you'll need to update the `scripts` in `package.json` like so:

```json filename="package.json"
{
  "scripts": {
    "dev": "node server.js",
    "build": "next build",
    "start": "NODE_ENV=production node server.js"
  }
}
```

***

The custom server uses the following import to connect the server with the Next.js application:

```js
const next = require('next')
const app = next({})
```

The above `next` import is a function that receives an object with the following options:

| Option         | Type               | Description                                                                                                   |
| -------------- | ------------------ | ------------------------------------------------------------------------------------------------------------- |
| `conf`         | `Object`           | The same object you would use in [next.config.js](/docs/pages/api-reference/next-config-js). Defaults to `{}` |
| `customServer` | `Boolean`          | (*Optional*) Set to false when the server was created by Next.js                                              |
| `dev`          | `Boolean`          | (*Optional*) Whether or not to launch Next.js in dev mode. Defaults to `false`                                |
| `dir`          | `String`           | (*Optional*) Location of the Next.js project. Defaults to `'.'`                                               |
| `quiet`        | `Boolean`          | (*Optional*) Hide error messages containing server information. Defaults to `false`                           |
| `hostname`     | `String`           | (*Optional*) The hostname the server is running behind                                                        |
| `port`         | `Number`           | (*Optional*) The port the server is running behind                                                            |
| `httpServer`   | `node:http#Server` | (*Optional*) The HTTP Server that Next.js is running behind                                                   |

The returned `app` can then be used to let Next.js handle requests as required.

## Disabling file-system routing

By default, `Next` will serve each file in the `pages` folder under a pathname matching the filename. If your project uses a custom server, this behavior may result in the same content being served from multiple paths, which can present problems with SEO and UX.

To disable this behavior and prevent routing based on files in `pages`, open `next.config.js` and disable the `useFileSystemPublicRoutes` config:

```js filename="next.config.js"
module.exports = {
  useFileSystemPublicRoutes: false,
}
```

> Note that `useFileSystemPublicRoutes` disables filename routes from SSR; client-side routing may still access those paths. When using this option, you should guard against navigation to routes you do not want programmatically.

> You may also wish to configure the client-side router to disallow client-side redirects to filename routes; for that refer to [`router.beforePopState`](/docs/pages/api-reference/functions/use-router#routerbeforepopstate).
---

For an index of all available documentation, see [/docs/14/llms.txt](/docs/14/llms.txt)