2018-04-21 04:34:02 +08:00
'use strict' ;
// Do this as the first thing so that any code reading it knows the right env.
process . env . BABEL _ENV = 'production' ;
process . env . NODE _ENV = 'production' ;
// Makes the script crash on unhandled rejections instead of silently
// ignoring them. In the future, promise rejections that are not handled will
// terminate the Node.js process with a non-zero exit code.
process . on ( 'unhandledRejection' , err => {
throw err ;
} ) ;
// Ensure environment variables are read.
require ( '../config/env' ) ;
2018-11-10 00:06:46 +08:00
const path = require ( 'path' ) ;
2019-04-28 10:19:34 +08:00
const chalk = require ( 'react-dev-utils/chalk' ) ;
2018-04-21 04:34:02 +08:00
const fs = require ( 'fs-extra' ) ;
const webpack = require ( 'webpack' ) ;
2019-04-28 10:19:34 +08:00
const configFactory = require ( '../config/webpack.config.demo' ) ;
2018-04-21 04:34:02 +08:00
const paths = require ( '../config/paths' ) ;
const checkRequiredFiles = require ( 'react-dev-utils/checkRequiredFiles' ) ;
const formatWebpackMessages = require ( 'react-dev-utils/formatWebpackMessages' ) ;
2018-11-10 00:06:46 +08:00
const printHostingInstructions = require ( 'react-dev-utils/printHostingInstructions' ) ;
2018-04-21 04:34:02 +08:00
const FileSizeReporter = require ( 'react-dev-utils/FileSizeReporter' ) ;
const printBuildError = require ( 'react-dev-utils/printBuildError' ) ;
const measureFileSizesBeforeBuild =
FileSizeReporter . measureFileSizesBeforeBuild ;
const printFileSizesAfterBuild = FileSizeReporter . printFileSizesAfterBuild ;
2018-11-10 00:06:46 +08:00
const useYarn = fs . existsSync ( paths . yarnLockFile ) ;
2018-04-21 04:34:02 +08:00
// These sizes are pretty large. We'll warn for bundles exceeding them.
const WARN _AFTER _BUNDLE _GZIP _SIZE = 512 * 1024 ;
const WARN _AFTER _CHUNK _GZIP _SIZE = 1024 * 1024 ;
2018-11-10 00:06:46 +08:00
const isInteractive = process . stdout . isTTY ;
2018-04-21 04:34:02 +08:00
// Warn and crash if required files are missing
2018-11-10 00:06:46 +08:00
if ( ! checkRequiredFiles ( [ paths . appHtml , paths . appDemoIndexJs ] ) ) {
2018-04-21 04:34:02 +08:00
process . exit ( 1 ) ;
}
2019-04-28 10:19:34 +08:00
// Generate configuration
const config = configFactory ( 'production' ) ;
2018-11-10 00:06:46 +08:00
// We require that you explicitly set browsers and do not fall back to
// browserslist defaults.
const { checkBrowsers } = require ( 'react-dev-utils/browsersHelper' ) ;
checkBrowsers ( paths . appPath , isInteractive )
. then ( ( ) => {
// First, read the current file sizes in build directory.
// This lets us display how much they changed later.
return measureFileSizesBeforeBuild ( paths . appDemo ) ;
} )
2018-04-21 04:34:02 +08:00
. then ( previousFileSizes => {
// Remove all content but keep the directory so that
// if you're in it, you don't end up in Trash
2018-11-10 00:06:46 +08:00
fs . emptyDirSync ( paths . appDemo ) ;
// Merge with the public folder
copyPublicFolder ( ) ;
2018-04-21 04:34:02 +08:00
// Start the webpack build
return build ( previousFileSizes ) ;
} )
. then (
( { stats , previousFileSizes , warnings } ) => {
if ( warnings . length ) {
console . log ( chalk . yellow ( 'Compiled with warnings.\n' ) ) ;
console . log ( warnings . join ( '\n\n' ) ) ;
console . log (
'\nSearch for the ' +
chalk . underline ( chalk . yellow ( 'keywords' ) ) +
' to learn more about each warning.'
) ;
console . log (
'To ignore, add ' +
chalk . cyan ( '// eslint-disable-next-line' ) +
' to the line before.\n'
) ;
} else {
console . log ( chalk . green ( 'Compiled successfully.\n' ) ) ;
}
console . log ( 'File sizes after gzip:\n' ) ;
printFileSizesAfterBuild (
stats ,
previousFileSizes ,
2018-11-10 00:06:46 +08:00
paths . appDemo ,
2018-04-21 04:34:02 +08:00
WARN _AFTER _BUNDLE _GZIP _SIZE ,
WARN _AFTER _CHUNK _GZIP _SIZE
) ;
console . log ( ) ;
2018-11-10 00:06:46 +08:00
const appPackage = require ( paths . appPackageJson ) ;
const publicUrl = paths . publicUrl ;
const publicPath = config . output . publicPath ;
const buildFolder = path . relative ( process . cwd ( ) , paths . appDemo ) ;
printHostingInstructions (
appPackage ,
publicUrl ,
publicPath ,
buildFolder ,
useYarn
) ;
2018-04-21 04:34:02 +08:00
} ,
err => {
2019-08-18 05:16:58 +08:00
console . error ( 'Failed to compile' ) ;
2018-04-21 04:34:02 +08:00
printBuildError ( err ) ;
process . exit ( 1 ) ;
}
2018-11-10 00:06:46 +08:00
)
. catch ( err => {
if ( err && err . message ) {
console . log ( err . message ) ;
}
process . exit ( 1 ) ;
} ) ;
2018-04-21 04:34:02 +08:00
// Create the production build and print the deployment instructions.
function build ( previousFileSizes ) {
2019-04-28 10:19:34 +08:00
// We used to support resolving modules according to `NODE_PATH`.
// This now has been deprecated in favor of jsconfig/tsconfig.json
// This lets you use absolute paths in imports inside large monorepos:
if ( process . env . NODE _PATH ) {
console . log (
chalk . yellow (
'Setting NODE_PATH to resolve modules absolutely has been deprecated in favor of setting baseUrl in jsconfig.json (or tsconfig.json if you are using TypeScript) and will be removed in a future major release of create-react-app.'
)
) ;
console . log ( ) ;
}
2018-11-10 00:06:46 +08:00
console . log ( 'Creating an optimized production build...' ) ;
2018-04-21 04:34:02 +08:00
2019-04-28 10:19:34 +08:00
const compiler = webpack ( config ) ;
2018-04-21 04:34:02 +08:00
return new Promise ( ( resolve , reject ) => {
compiler . run ( ( err , stats ) => {
2018-11-10 00:06:46 +08:00
let messages ;
2018-04-21 04:34:02 +08:00
if ( err ) {
2018-11-10 00:06:46 +08:00
if ( ! err . message ) {
return reject ( err ) ;
}
messages = formatWebpackMessages ( {
errors : [ err . message ] ,
warnings : [ ] ,
} ) ;
} else {
messages = formatWebpackMessages (
stats . toJson ( { all : false , warnings : true , errors : true } )
) ;
2018-04-21 04:34:02 +08:00
}
if ( messages . errors . length ) {
// Only keep the first error. Others are often indicative
// of the same problem, but confuse the reader with noise.
if ( messages . errors . length > 1 ) {
messages . errors . length = 1 ;
}
return reject ( new Error ( messages . errors . join ( '\n\n' ) ) ) ;
}
if (
process . env . CI &&
( typeof process . env . CI !== 'string' ||
process . env . CI . toLowerCase ( ) !== 'false' ) &&
messages . warnings . length
) {
console . log (
chalk . yellow (
'\nTreating warnings as errors because process.env.CI = true.\n' +
'Most CI servers set it automatically.\n'
)
) ;
return reject ( new Error ( messages . warnings . join ( '\n\n' ) ) ) ;
}
2018-11-10 00:06:46 +08:00
2019-04-28 10:19:34 +08:00
return resolve ( {
2018-04-21 04:34:02 +08:00
stats ,
previousFileSizes ,
warnings : messages . warnings ,
2019-04-28 10:19:34 +08:00
} ) ;
2018-04-21 04:34:02 +08:00
} ) ;
} ) ;
}
2018-11-10 00:06:46 +08:00
function copyPublicFolder ( ) {
fs . copySync ( paths . appPublic , paths . appDemo , {
dereference : true ,
filter : file => file !== paths . appHtml ,
} ) ;
}