2016-10-06 11:05:52 +00:00
|
|
|
#!/usr/bin/env node
|
2016-10-17 01:04:35 +00:00
|
|
|
import { resolve, join } from 'path'
|
2018-12-15 21:55:59 +00:00
|
|
|
import arg from 'arg'
|
2018-09-14 20:44:17 +00:00
|
|
|
import { existsSync } from 'fs'
|
2018-09-27 19:10:53 +00:00
|
|
|
import startServer from '../server/lib/start-server'
|
|
|
|
import { printAndExit } from '../server/lib/utils'
|
2016-10-06 11:05:52 +00:00
|
|
|
|
2018-12-15 21:55:59 +00:00
|
|
|
const args = arg({
|
|
|
|
// Types
|
|
|
|
'--help': Boolean,
|
|
|
|
'--port': Number,
|
|
|
|
'--hostname': String,
|
|
|
|
|
|
|
|
// Aliases
|
|
|
|
'-h': '--help',
|
|
|
|
'-p': '--port',
|
2018-12-31 13:44:27 +00:00
|
|
|
'-H': '--hostname',
|
2016-10-06 11:05:52 +00:00
|
|
|
})
|
|
|
|
|
2018-12-15 21:55:59 +00:00
|
|
|
if (args['--help']) {
|
2018-12-31 13:44:27 +00:00
|
|
|
// tslint:disable-next-line
|
2016-12-18 20:28:34 +00:00
|
|
|
console.log(`
|
|
|
|
Description
|
|
|
|
Starts the application in development mode (hot-code reloading, error
|
|
|
|
reporting, etc)
|
|
|
|
|
|
|
|
Usage
|
|
|
|
$ next dev <dir> -p <port number>
|
|
|
|
|
2017-04-06 10:09:26 +00:00
|
|
|
<dir> represents where the compiled folder should go.
|
|
|
|
If no directory is provided, the folder will be created in the current directory.
|
|
|
|
You can set a custom folder in config https://github.com/zeit/next.js#custom-configuration.
|
2016-12-18 20:28:34 +00:00
|
|
|
|
|
|
|
Options
|
|
|
|
--port, -p A port number on which to start the application
|
2017-07-08 20:14:58 +00:00
|
|
|
--hostname, -H Hostname on which to start the application
|
2017-01-15 01:31:09 +00:00
|
|
|
--help, -h Displays this message
|
2016-12-18 20:28:34 +00:00
|
|
|
`)
|
|
|
|
process.exit(0)
|
|
|
|
}
|
|
|
|
|
2018-12-15 21:55:59 +00:00
|
|
|
const dir = resolve(args._[0] || '.')
|
2016-10-06 11:05:52 +00:00
|
|
|
|
2017-01-05 21:00:55 +00:00
|
|
|
// Check if pages dir exists and warn if not
|
2017-01-15 12:15:06 +00:00
|
|
|
if (!existsSync(dir)) {
|
|
|
|
printAndExit(`> No such directory exists as the project root: ${dir}`)
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!existsSync(join(dir, 'pages'))) {
|
|
|
|
if (existsSync(join(dir, '..', 'pages'))) {
|
|
|
|
printAndExit('> No `pages` directory found. Did you mean to run `next` in the parent (`../`) directory?')
|
2016-10-17 01:04:35 +00:00
|
|
|
}
|
2017-01-15 12:15:06 +00:00
|
|
|
|
|
|
|
printAndExit('> Couldn\'t find a `pages` directory. Please create one under the project root')
|
|
|
|
}
|
2017-01-05 21:00:55 +00:00
|
|
|
|
2018-12-15 21:55:59 +00:00
|
|
|
const port = args['--port'] || 3000
|
|
|
|
startServer({dir, dev: true}, port, args['--hostname'])
|
Improve dev experience by listening faster (#5902)
As I detailed in [this thread on Spectrum](https://spectrum.chat/?t=3df7b1fb-7331-4ca4-af35-d9a8b1cacb2c), the dev experience would be a lot nicer if the server started listening as soon as possible, before the slow initialization steps. That way, instead of manually polling the dev URL until the server's up (this can take a long time!), I can open it right away and the responses will be delivered when the dev server is done initializing.
This makes a few changes to the dev server:
* Move `HotReloader` creation to `prepare`. Ideally, more things (from the non-dev `Server`) would be moved to a later point as well, because creating `next({ ... })` is quite slow.
* In `run`, wait for a promise to resolve before doing anything. This promise automatically gets resolved whenever `prepare` finishes successfully.
And the `next dev` and `next start` scripts:
* Since we want to log that the server is ready/listening before the intensive build process kicks off, we return the app instance from `startServer` and the scripts call `app.prepare()`.
This should all be backwards compatible, including with all existing custom server recommendations that essentially say `app.prepare().then(listen)`. But now, we could make an even better recommendation: start listening right away, then call `app.prepare()` in the `listen` callback. Users would be free to make that change and get better DX.
Try it and I doubt you'll want to go back to the old way. :)
2018-12-17 11:09:44 +00:00
|
|
|
.then(async (app) => {
|
2018-12-31 13:44:27 +00:00
|
|
|
// tslint:disable-next-line
|
2018-12-15 21:55:59 +00:00
|
|
|
console.log(`> Ready on http://${args['--hostname'] || 'localhost'}:${port}`)
|
Improve dev experience by listening faster (#5902)
As I detailed in [this thread on Spectrum](https://spectrum.chat/?t=3df7b1fb-7331-4ca4-af35-d9a8b1cacb2c), the dev experience would be a lot nicer if the server started listening as soon as possible, before the slow initialization steps. That way, instead of manually polling the dev URL until the server's up (this can take a long time!), I can open it right away and the responses will be delivered when the dev server is done initializing.
This makes a few changes to the dev server:
* Move `HotReloader` creation to `prepare`. Ideally, more things (from the non-dev `Server`) would be moved to a later point as well, because creating `next({ ... })` is quite slow.
* In `run`, wait for a promise to resolve before doing anything. This promise automatically gets resolved whenever `prepare` finishes successfully.
And the `next dev` and `next start` scripts:
* Since we want to log that the server is ready/listening before the intensive build process kicks off, we return the app instance from `startServer` and the scripts call `app.prepare()`.
This should all be backwards compatible, including with all existing custom server recommendations that essentially say `app.prepare().then(listen)`. But now, we could make an even better recommendation: start listening right away, then call `app.prepare()` in the `listen` callback. Users would be free to make that change and get better DX.
Try it and I doubt you'll want to go back to the old way. :)
2018-12-17 11:09:44 +00:00
|
|
|
await app.prepare()
|
2018-03-27 18:11:03 +00:00
|
|
|
})
|
|
|
|
.catch((err) => {
|
|
|
|
if (err.code === 'EADDRINUSE') {
|
2018-12-15 21:55:59 +00:00
|
|
|
let errorMessage = `Port ${port} is already in use.`
|
2018-03-27 18:11:03 +00:00
|
|
|
const pkgAppPath = require('find-up').sync('package.json', {
|
2018-12-31 13:44:27 +00:00
|
|
|
cwd: dir,
|
2018-03-27 18:11:03 +00:00
|
|
|
})
|
2018-09-14 20:44:17 +00:00
|
|
|
const appPackage = require(pkgAppPath)
|
|
|
|
if (appPackage.scripts) {
|
2018-12-31 13:44:27 +00:00
|
|
|
const nextScript = Object.entries(appPackage.scripts).find((scriptLine) => scriptLine[1] === 'next')
|
2018-09-14 20:44:17 +00:00
|
|
|
if (nextScript) {
|
|
|
|
errorMessage += `\nUse \`npm run ${nextScript[0]} -- -p <some other port>\`.`
|
|
|
|
}
|
|
|
|
}
|
2018-12-31 13:44:27 +00:00
|
|
|
// tslint:disable-next-line
|
2018-03-27 18:11:03 +00:00
|
|
|
console.error(errorMessage)
|
|
|
|
} else {
|
2018-12-31 13:44:27 +00:00
|
|
|
// tslint:disable-next-line
|
2018-03-27 18:11:03 +00:00
|
|
|
console.error(err)
|
|
|
|
}
|
|
|
|
process.nextTick(() => process.exit(1))
|
|
|
|
})
|