From b284377ebe7fe7de86c7fd60c43457ce92069d32 Mon Sep 17 00:00:00 2001 From: Dan Abramov Date: Mon, 24 Oct 2016 17:53:05 +0100 Subject: [PATCH 01/19] Encourage people to try recent npm --- ISSUE_TEMPLATE.md | 16 ++++++++++++++++ 1 file changed, 16 insertions(+) diff --git a/ISSUE_TEMPLATE.md b/ISSUE_TEMPLATE.md index 2bcdd77dc36..e85da85b4c7 100644 --- a/ISSUE_TEMPLATE.md +++ b/ISSUE_TEMPLATE.md @@ -1,5 +1,21 @@ If you are reporting a bug, please fill in below. Otherwise feel free to remove this template entirely. +### Can you reproduce the problem with latest npm? + +Run: + +``` +npm install -g npm@latest + +cd your_project_directory +rm -rf node_modules +npm install +``` + +and try to reproduce the issue again. + +Can you still reproduce it? + ### Description What are you reporting? From 91e616e4dc8f191d5ae0d15a1b9ad1bea6c9b3e2 Mon Sep 17 00:00:00 2001 From: Dan Abramov Date: Tue, 25 Oct 2016 12:01:58 +0100 Subject: [PATCH 02/19] Point people to npm Windows instructions --- ISSUE_TEMPLATE.md | 8 ++++++-- 1 file changed, 6 insertions(+), 2 deletions(-) diff --git a/ISSUE_TEMPLATE.md b/ISSUE_TEMPLATE.md index e85da85b4c7..374dbf17958 100644 --- a/ISSUE_TEMPLATE.md +++ b/ISSUE_TEMPLATE.md @@ -2,7 +2,11 @@ If you are reporting a bug, please fill in below. Otherwise feel free to remove ### Can you reproduce the problem with latest npm? -Run: +Many errors, especially related to "missing modules", are due to npm bugs. + +If you're using Windows, [follow these instructions to update npm](https://github.com/npm/npm/wiki/Troubleshooting#upgrading-on-windows). + +If you're using OS X or Linux, run this to update npm: ``` npm install -g npm@latest @@ -12,7 +16,7 @@ rm -rf node_modules npm install ``` -and try to reproduce the issue again. +Then try to reproduce the issue again. Can you still reproduce it? From 75d158528c92f5ac3cdefd4f545015bde004706b Mon Sep 17 00:00:00 2001 From: Dan Abramov Date: Tue, 25 Oct 2016 23:07:28 +0100 Subject: [PATCH 03/19] Add next.js to Alternatives --- README.md | 1 + 1 file changed, 1 insertion(+) diff --git a/README.md b/README.md index 334dd477e85..b7e1f1f45db 100644 --- a/README.md +++ b/README.md @@ -205,6 +205,7 @@ Some of the more popular and actively maintained ones are: * [insin/nwb](https://github.com/insin/nwb) * [mozilla/neo](https://github.com/mozilla/neo) * [NYTimes/kyt](https://github.com/NYTimes/kyt) +* [zeit/next.js](https://github.com/zeit/next.js) Notable alternatives also include: From ca443abe62999df0443b40204c690e0a24a2152a Mon Sep 17 00:00:00 2001 From: Fatih Date: Fri, 28 Oct 2016 15:41:51 +0300 Subject: [PATCH 04/19] Enable compression on webpack-dev-server (#966) (#968) --- packages/react-scripts/scripts/start.js | 2 ++ 1 file changed, 2 insertions(+) diff --git a/packages/react-scripts/scripts/start.js b/packages/react-scripts/scripts/start.js index 8723c281637..8a115dd8e27 100644 --- a/packages/react-scripts/scripts/start.js +++ b/packages/react-scripts/scripts/start.js @@ -195,6 +195,8 @@ function addMiddleware(devServer) { function runDevServer(host, port, protocol) { var devServer = new WebpackDevServer(compiler, { + // Enable gzip compression of generated files. + compress: true, // Silence WebpackDevServer's own logs since they're generally not useful. // It will still show compile warnings and errors with this setting. clientLogLevel: 'none', From 91c86502a4cfef1e494909b51c98a39a2d93062c Mon Sep 17 00:00:00 2001 From: Swizec Teller Date: Fri, 28 Oct 2016 05:42:51 -0700 Subject: [PATCH 05/19] Gently nudge users towards https by default (#974) gh-pages supports https. It's important for prominent help files to encourage best practices. --- packages/react-scripts/template/README.md | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/packages/react-scripts/template/README.md b/packages/react-scripts/template/README.md index 6f13b12d2e0..6bb93795bad 100644 --- a/packages/react-scripts/template/README.md +++ b/packages/react-scripts/template/README.md @@ -881,7 +881,7 @@ This will let Create React App correctly infer the root path to use in the gener Open your `package.json` and add a `homepage` field: ```js - "homepage": "http://myusername.github.io/my-app", + "homepage": "https://myusername.github.io/my-app", ``` **The above step is important!**
@@ -889,7 +889,7 @@ Create React App uses the `homepage` field to determine the root URL in the buil Now, whenever you run `npm run build`, you will see a cheat sheet with instructions on how to deploy to GitHub pages. -To publish it at [http://myusername.github.io/my-app](http://myusername.github.io/my-app), run: +To publish it at [https://myusername.github.io/my-app](https://myusername.github.io/my-app), run: ```sh npm install --save-dev gh-pages From 0bd593baa777606ffb72f63eab9f5fd491062007 Mon Sep 17 00:00:00 2001 From: Sandro Padin Date: Fri, 28 Oct 2016 05:45:50 -0700 Subject: [PATCH 06/19] Catch and noop call to open web browser. (#964) Running `create-react-app` in a Docker container causes an unhandled rejection error in nodejs > 6.5 because the `opn` module tries to open a web browser in a system that doesn't have one. I figured this error could be safely ignored. --- packages/react-dev-utils/openBrowser.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/react-dev-utils/openBrowser.js b/packages/react-dev-utils/openBrowser.js index 76b33a5924a..bee85a7d299 100644 --- a/packages/react-dev-utils/openBrowser.js +++ b/packages/react-dev-utils/openBrowser.js @@ -28,7 +28,7 @@ function openBrowser(url) { // Fallback to opn // (It will always open new tab) try { - opn(url); + opn(url).catch(() => {}); // Prevent `unhandledRejection` error. return true; } catch (err) { return false; From a0efbebee2671d1f11bc0a1936b84e45ab322c19 Mon Sep 17 00:00:00 2001 From: Patrick Mackinder Date: Fri, 28 Oct 2016 13:47:08 +0100 Subject: [PATCH 07/19] Add collectCoverageFrom option to collect coverage on files without any tests. (#961) --- packages/react-scripts/utils/createJestConfig.js | 1 + 1 file changed, 1 insertion(+) diff --git a/packages/react-scripts/utils/createJestConfig.js b/packages/react-scripts/utils/createJestConfig.js index 39c864ab8f4..17d414fc594 100644 --- a/packages/react-scripts/utils/createJestConfig.js +++ b/packages/react-scripts/utils/createJestConfig.js @@ -18,6 +18,7 @@ module.exports = (resolve, rootDir, isEjecting) => { const setupTestsFile = pathExists.sync(paths.testsSetup) ? '/src/setupTests.js' : undefined; const config = { + collectCoverageFrom: ['src/**/*.{js,jsx}'], moduleFileExtensions: ['jsx', 'js', 'json'], moduleNameMapper: { '^.+\\.(jpg|jpeg|png|gif|eot|otf|webp|svg|ttf|woff|woff2|mp4|webm|wav|mp3|m4a|aac|oga)$': resolve('config/jest/FileStub.js'), From fcda24eee879460095e35e6e5b66269cca8b8094 Mon Sep 17 00:00:00 2001 From: David Ernst Date: Fri, 28 Oct 2016 05:51:06 -0700 Subject: [PATCH 08/19] Always build before deploying to gh-pages (#959) * Always build before deploying to gh-pages * Add line to gh-pages deploy docs about CNAME file * Remove spaces in npm run command for Windows * Grammar nit * Minor tweaks --- packages/react-scripts/template/README.md | 8 ++++++-- 1 file changed, 6 insertions(+), 2 deletions(-) diff --git a/packages/react-scripts/template/README.md b/packages/react-scripts/template/README.md index 6bb93795bad..788f77e7805 100644 --- a/packages/react-scripts/template/README.md +++ b/packages/react-scripts/template/README.md @@ -887,7 +887,7 @@ Open your `package.json` and add a `homepage` field: **The above step is important!**
Create React App uses the `homepage` field to determine the root URL in the built HTML file. -Now, whenever you run `npm run build`, you will see a cheat sheet with instructions on how to deploy to GitHub pages. +Now, whenever you run `npm run build`, you will see a cheat sheet with instructions on how to deploy to GitHub Pages. To publish it at [https://myusername.github.io/my-app](https://myusername.github.io/my-app), run: @@ -901,16 +901,20 @@ Add the following script in your `package.json`: // ... "scripts": { // ... - "deploy": "gh-pages -d build" + "deploy": "npm run build&&gh-pages -d build" } ``` +(Note: the lack of whitespace is intentional.) + Then run: ```sh npm run deploy ``` +You can configure a custom domain with GitHub Pages by adding a `CNAME` file to the `public/` folder. + Note that GitHub Pages doesn't support routers that use the HTML5 `pushState` history API under the hood (for example, React Router using `browserHistory`). This is because when there is a fresh page load for a url like `http://user.github.io/todomvc/todos/42`, where `/todos/42` is a frontend route, the GitHub Pages server returns 404 because it knows nothing of `/todos/42`. If you want to add a router to a project hosted on GitHub Pages, here are a couple of solutions: * You could switch from using HTML5 history API to routing with hashes. If you use React Router, you can switch to `hashHistory` for this effect, but the URL will be longer and more verbose (for example, `http://user.github.io/todomvc/#/todos/42?_k=yknaj`). [Read more](https://github.com/reactjs/react-router/blob/master/docs/guides/Histories.md#histories) about different history implementations in React Router. * Alternatively, you can use a trick to teach GitHub Pages to handle 404 by redirecting to your `index.html` page with a special redirect parameter. You would need to add a `404.html` file with the redirection code to the `build` folder before deploying your project, and you’ll need to add code handling the redirect parameter to `index.html`. You can find a detailed explanation of this technique [in this guide](https://github.com/rafrex/spa-github-pages). From 8a5c50d97a3f41eceb8f237f1ffb8e18fd21ee16 Mon Sep 17 00:00:00 2001 From: Vadzim Date: Fri, 28 Oct 2016 15:51:56 +0300 Subject: [PATCH 09/19] Fixes https://github.com/facebookincubator/create-react-app/issues/952 (#953) --- packages/eslint-config-react-app/index.js | 2 +- packages/react-scripts/utils/createJestConfig.js | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/packages/eslint-config-react-app/index.js b/packages/eslint-config-react-app/index.js index f2f79a2dcc1..6df5bfdd93b 100644 --- a/packages/eslint-config-react-app/index.js +++ b/packages/eslint-config-react-app/index.js @@ -44,7 +44,7 @@ module.exports = { settings: { 'import/ignore': [ 'node_modules', - '\\.(json|css|jpg|jpeg|png|gif|eot|otf|webp|svg|ttf|woff|woff2|mp4|webm|wav|mp3|m4a|aac|oga)$', + '\\.(json|css|ico|jpg|jpeg|png|gif|eot|otf|webp|svg|ttf|woff|woff2|mp4|webm|wav|mp3|m4a|aac|oga)$', ], 'import/extensions': ['.js'], 'import/resolver': { diff --git a/packages/react-scripts/utils/createJestConfig.js b/packages/react-scripts/utils/createJestConfig.js index 17d414fc594..df0238f2587 100644 --- a/packages/react-scripts/utils/createJestConfig.js +++ b/packages/react-scripts/utils/createJestConfig.js @@ -21,7 +21,7 @@ module.exports = (resolve, rootDir, isEjecting) => { collectCoverageFrom: ['src/**/*.{js,jsx}'], moduleFileExtensions: ['jsx', 'js', 'json'], moduleNameMapper: { - '^.+\\.(jpg|jpeg|png|gif|eot|otf|webp|svg|ttf|woff|woff2|mp4|webm|wav|mp3|m4a|aac|oga)$': resolve('config/jest/FileStub.js'), + '^.+\\.(ico|jpg|jpeg|png|gif|eot|otf|webp|svg|ttf|woff|woff2|mp4|webm|wav|mp3|m4a|aac|oga)$': resolve('config/jest/FileStub.js'), '^.+\\.css$': resolve('config/jest/CSSStub.js') }, setupFiles: [resolve('config/polyfills.js')], From 7ce4b6e0e436feff0a38cebb123eefdf6a6dc399 Mon Sep 17 00:00:00 2001 From: Alice Rose Date: Fri, 28 Oct 2016 14:54:56 +0200 Subject: [PATCH 10/19] Check for presence of folders before continuing eject. Closes #939. (#951) --- packages/react-scripts/scripts/eject.js | 38 ++++++++++++++++--------- 1 file changed, 24 insertions(+), 14 deletions(-) diff --git a/packages/react-scripts/scripts/eject.js b/packages/react-scripts/scripts/eject.js index d14aec6abef..dbd4d64e4d5 100644 --- a/packages/react-scripts/scripts/eject.js +++ b/packages/react-scripts/scripts/eject.js @@ -30,6 +30,25 @@ prompt( var ownPath = path.join(__dirname, '..'); var appPath = path.join(ownPath, '..', '..'); + + function verifyAbsent(file) { + if (fs.existsSync(path.join(appPath, file))) { + console.error( + '`' + file + '` already exists in your app folder. We cannot ' + + 'continue as you would lose all the changes in that file or directory. ' + + 'Please move or delete it (maybe make a copy for backup) and run this ' + + 'command again.' + ); + process.exit(1); + } + } + + var folders = [ + 'config', + path.join('config', 'jest'), + 'scripts' + ]; + var files = [ path.join('config', 'env.js'), path.join('config', 'paths.js'), @@ -44,22 +63,13 @@ prompt( ]; // Ensure that the app folder is clean and we won't override any files - files.forEach(function(file) { - if (fs.existsSync(path.join(appPath, file))) { - console.error( - '`' + file + '` already exists in your app folder. We cannot ' + - 'continue as you would lose all the changes in that file or directory. ' + - 'Please delete it (maybe make a copy for backup) and run this ' + - 'command again.' - ); - process.exit(1); - } - }); + folders.forEach(verifyAbsent); + files.forEach(verifyAbsent); // Copy the files over - fs.mkdirSync(path.join(appPath, 'config')); - fs.mkdirSync(path.join(appPath, 'config', 'jest')); - fs.mkdirSync(path.join(appPath, 'scripts')); + folders.forEach(function(folder) { + fs.mkdirSync(path.join(appPath, folder)) + }); console.log(); console.log(cyan('Copying files into ' + appPath)); From 6bda4b972602ca339059a5a5c829f41f852aed69 Mon Sep 17 00:00:00 2001 From: Ville Immonen Date: Sat, 29 Oct 2016 21:00:16 +0200 Subject: [PATCH 11/19] Remove custom babel-loader cache dir config (#983) Upgrade `babel-loader` and remove the cache directory configuration that was added in #620. `babel-loader` now uses the `./node_modules/.cache/babel-loader` directory by default, so the custom config is no longer needed. --- packages/react-scripts/config/webpack.config.dev.js | 10 +++------- packages/react-scripts/package.json | 3 +-- 2 files changed, 4 insertions(+), 9 deletions(-) diff --git a/packages/react-scripts/config/webpack.config.dev.js b/packages/react-scripts/config/webpack.config.dev.js index d875c63e8d9..c2b544cca54 100644 --- a/packages/react-scripts/config/webpack.config.dev.js +++ b/packages/react-scripts/config/webpack.config.dev.js @@ -12,7 +12,6 @@ var path = require('path'); var autoprefixer = require('autoprefixer'); var webpack = require('webpack'); -var findCacheDir = require('find-cache-dir'); var HtmlWebpackPlugin = require('html-webpack-plugin'); var CaseSensitivePathsPlugin = require('case-sensitive-paths-webpack-plugin'); var InterpolateHtmlPlugin = require('react-dev-utils/InterpolateHtmlPlugin'); @@ -122,12 +121,9 @@ module.exports = { presets: [require.resolve('babel-preset-react-app')], // @remove-on-eject-end // This is a feature of `babel-loader` for webpack (not Babel itself). - // It enables caching results in ./node_modules/.cache/react-scripts/ - // directory for faster rebuilds. We use findCacheDir() because of: - // https://github.com/facebookincubator/create-react-app/issues/483 - cacheDirectory: findCacheDir({ - name: 'react-scripts' - }) + // It enables caching results in ./node_modules/.cache/babel-loader/ + // directory for faster rebuilds. + cacheDirectory: true } }, // "postcss" loader applies autoprefixer to our CSS. diff --git a/packages/react-scripts/package.json b/packages/react-scripts/package.json index 58bb4ab1d06..b4cad963979 100644 --- a/packages/react-scripts/package.json +++ b/packages/react-scripts/package.json @@ -27,7 +27,7 @@ "babel-core": "6.17.0", "babel-eslint": "7.0.0", "babel-jest": "16.0.0", - "babel-loader": "6.2.5", + "babel-loader": "6.2.7", "babel-preset-react-app": "^1.0.0", "case-sensitive-paths-webpack-plugin": "1.1.4", "chalk": "1.1.3", @@ -46,7 +46,6 @@ "extract-text-webpack-plugin": "1.0.1", "file-loader": "0.9.0", "filesize": "3.3.0", - "find-cache-dir": "0.1.1", "fs-extra": "0.30.0", "gzip-size": "3.0.0", "html-webpack-plugin": "2.24.0", From e3b69661d9b056277b59aa1c4d358009a1f2f3a6 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Einar=20L=C3=B6ve?= Date: Mon, 31 Oct 2016 12:45:44 +0100 Subject: [PATCH 12/19] Allow webpack 2 as peerDependency in react-dev-utils (#963) * Allow webpack 2 as peerDependency * Remove webpack as peer dependency --- packages/react-dev-utils/package.json | 3 --- 1 file changed, 3 deletions(-) diff --git a/packages/react-dev-utils/package.json b/packages/react-dev-utils/package.json index 14c860999ae..1fe5c556c79 100644 --- a/packages/react-dev-utils/package.json +++ b/packages/react-dev-utils/package.json @@ -29,8 +29,5 @@ "opn": "4.0.2", "sockjs-client": "1.0.3", "strip-ansi": "3.0.1" - }, - "peerDependencies": { - "webpack": "^1.13.2" } } From b855cc368c346fb4cd71d1e3fea9218ac0553c0d Mon Sep 17 00:00:00 2001 From: Kyle Mathews Date: Mon, 31 Oct 2016 14:33:03 -0700 Subject: [PATCH 13/19] Add Gatsby to alternatives (#995) --- README.md | 1 + 1 file changed, 1 insertion(+) diff --git a/README.md b/README.md index b7e1f1f45db..df0046aa39a 100644 --- a/README.md +++ b/README.md @@ -206,6 +206,7 @@ Some of the more popular and actively maintained ones are: * [mozilla/neo](https://github.com/mozilla/neo) * [NYTimes/kyt](https://github.com/NYTimes/kyt) * [zeit/next.js](https://github.com/zeit/next.js) +* [gatsbyjs/gatsby](https://github.com/gatsbyjs/gatsby) Notable alternatives also include: From 79160b858a8496a93abbbe6254fb4008e22a6b89 Mon Sep 17 00:00:00 2001 From: Leo Wong Date: Tue, 1 Nov 2016 21:56:29 +0800 Subject: [PATCH 14/19] Remove redundant `function` from export statement (#996) --- packages/react-scripts/template/README.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/react-scripts/template/README.md b/packages/react-scripts/template/README.md index 788f77e7805..3203c5fcbb9 100644 --- a/packages/react-scripts/template/README.md +++ b/packages/react-scripts/template/README.md @@ -320,7 +320,7 @@ function Header() { return Logo; } -export default function Header; +export default Header; ``` This ensures that when the project is built, Webpack will correctly move the images into the build folder, and provide us with correct paths. From da1d534a6b25128f4e96da2c741041c3e220f89a Mon Sep 17 00:00:00 2001 From: Vesa Laakso Date: Thu, 17 Nov 2016 14:48:22 +0200 Subject: [PATCH 15/19] Update babel-present-env and use node: 'current' as target (#1051) * Update babel-preset-env to 0.0.8 Changes between 0.0.6 and 0.0.8 should be backwards compatible: https://github.com/babel/babel-preset-env/blob/master/CHANGELOG.md * Use `node: 'current'` as target for babel-preset-env This replaces the hand-rolled node version setup with a new feature that was introduced in babel-preset-env@v0.0.7 https://github.com/babel/babel-preset-env/blob/v0.0.7/CHANGELOG.md --- packages/babel-preset-react-app/index.js | 2 +- packages/babel-preset-react-app/package.json | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/packages/babel-preset-react-app/index.js b/packages/babel-preset-react-app/index.js index d6e9c4519d4..7af36414246 100644 --- a/packages/babel-preset-react-app/index.js +++ b/packages/babel-preset-react-app/index.js @@ -68,7 +68,7 @@ if (env === 'test') { // ES features necessary for user's Node version [require('babel-preset-env').default, { targets: { - node: parseFloat(process.versions.node), + node: 'current', }, }], // JSX, Flow diff --git a/packages/babel-preset-react-app/package.json b/packages/babel-preset-react-app/package.json index bba2df1e0b1..6ddae417bc2 100644 --- a/packages/babel-preset-react-app/package.json +++ b/packages/babel-preset-react-app/package.json @@ -20,7 +20,7 @@ "babel-plugin-transform-react-jsx-source": "6.9.0", "babel-plugin-transform-regenerator": "6.16.1", "babel-plugin-transform-runtime": "6.15.0", - "babel-preset-env": "0.0.6", + "babel-preset-env": "0.0.8", "babel-preset-latest": "6.16.0", "babel-preset-react": "6.16.0", "babel-runtime": "6.11.6" From 80fe767bf253227a4633c8ec3242f8ac3d90b566 Mon Sep 17 00:00:00 2001 From: Sathish Date: Thu, 17 Nov 2016 18:24:12 +0530 Subject: [PATCH 16/19] Clears the usage of react-jsx-source & react-jsx-self (#992) Explain the usage of react-jsx-source & react-jsx-self --- packages/babel-preset-react-app/index.js | 7 ++++++- 1 file changed, 6 insertions(+), 1 deletion(-) diff --git a/packages/babel-preset-react-app/index.js b/packages/babel-preset-react-app/index.js index 7af36414246..2aa1641ce35 100644 --- a/packages/babel-preset-react-app/index.js +++ b/packages/babel-preset-react-app/index.js @@ -54,6 +54,12 @@ if (env !== 'development' && env !== 'test' && env !== 'production') { } if (env === 'development' || env === 'test') { + // The following two plugins are currently necessary to make React warnings + // include more valuable information. They are included here because they are + // currently not enabled in babel-preset-react. See the below threads for more info: + // https://github.com/babel/babel/issues/4702 + // https://github.com/babel/babel/pull/3540#issuecomment-228673661 + // https://github.com/facebookincubator/create-react-app/issues/989 plugins.push.apply(plugins, [ // Adds component stack to warning messages require.resolve('babel-plugin-transform-react-jsx-source'), @@ -99,4 +105,3 @@ if (env === 'test') { // ]); } } - From 4a7f78ed66c60603441187cc5ce696db369c9c6a Mon Sep 17 00:00:00 2001 From: Vesa Laakso Date: Thu, 17 Nov 2016 18:14:36 +0200 Subject: [PATCH 17/19] Remove unnecessary transform plugins for object spread to work (#1052) * Update `babel-plugin-transform-object-rest-spread` to v6.19.0 The `babel-plugin-transform-object-rest-spread` v6.19.0 update will allow us to remove the `babel-plugin-transform-es2015-destructuring` and `babel-plugin-transform-es2015-parameters` as the object rest spread transform will now work standalone and not require additional tranforms * Remove unnecessary babel transform plugins from babel-preset-react-app The `babel-plugin-transform-object-rest-spread` v6.19.0 update makes these plugins unnecessary, as v6.19.0 can be used stand-alone --- packages/babel-preset-react-app/index.js | 10 +--------- packages/babel-preset-react-app/package.json | 4 +--- 2 files changed, 2 insertions(+), 12 deletions(-) diff --git a/packages/babel-preset-react-app/index.js b/packages/babel-preset-react-app/index.js index 2aa1641ce35..a2639c6d4cf 100644 --- a/packages/babel-preset-react-app/index.js +++ b/packages/babel-preset-react-app/index.js @@ -27,15 +27,7 @@ const plugins = [ regenerator: true, // Resolve the Babel runtime relative to the config. moduleName: path.dirname(require.resolve('babel-runtime/package')) - }], - // The following two plugins are currently necessary to get - // babel-preset-env to work with rest/spread. More info here: - // https://github.com/babel/babel-preset-env#caveats - // https://github.com/babel/babel/issues/4074 - // const { a, ...z } = obj; - require.resolve('babel-plugin-transform-es2015-destructuring'), - // const fn = ({ a, ...otherProps }) => otherProps; - require.resolve('babel-plugin-transform-es2015-parameters') + }] ]; // This is similar to how `env` works in Babel: diff --git a/packages/babel-preset-react-app/package.json b/packages/babel-preset-react-app/package.json index 6ddae417bc2..13f00ed227d 100644 --- a/packages/babel-preset-react-app/package.json +++ b/packages/babel-preset-react-app/package.json @@ -12,9 +12,7 @@ ], "dependencies": { "babel-plugin-transform-class-properties": "6.16.0", - "babel-plugin-transform-es2015-destructuring": "6.16.0", - "babel-plugin-transform-es2015-parameters": "6.17.0", - "babel-plugin-transform-object-rest-spread": "6.16.0", + "babel-plugin-transform-object-rest-spread": "6.19.0", "babel-plugin-transform-react-constant-elements": "6.9.1", "babel-plugin-transform-react-jsx-self": "6.11.0", "babel-plugin-transform-react-jsx-source": "6.9.0", From b9c9aeda0352a18ff8db1d4f9cfd5a32004f1198 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Einar=20L=C3=B6ve?= Date: Thu, 17 Nov 2016 20:37:20 +0100 Subject: [PATCH 18/19] Fix chrome tab reuse (#1035) * Correctly checks site url to tab url in reuse check * Bring chrome to foreground focused after tab reuse --- packages/react-dev-utils/openChrome.applescript | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/packages/react-dev-utils/openChrome.applescript b/packages/react-dev-utils/openChrome.applescript index 4dfec4a2657..b36b70f6cfc 100644 --- a/packages/react-dev-utils/openChrome.applescript +++ b/packages/react-dev-utils/openChrome.applescript @@ -23,7 +23,7 @@ on run argv set theTabIndex to 0 repeat with theTab in every tab of theWindow set theTabIndex to theTabIndex + 1 - if theTab's URL is theURL then + if theTab's URL as string contains theURL then set found to true exit repeat end if @@ -38,6 +38,7 @@ on run argv tell theTab to reload set index of theWindow to 1 set theWindow's active tab index to theTabIndex + tell theWindow to activate else tell window 1 activate From bcc469c9a5c7916ec10786f133769cdda2c80188 Mon Sep 17 00:00:00 2001 From: Ville Immonen Date: Thu, 17 Nov 2016 22:55:00 +0200 Subject: [PATCH 19/19] Support Yarn (#898) In the `create-react-app` command, try to install packages using Yarn. If Yarn is not installed, use npm instead. In `react-scripts`, detect if the project is using Yarn by checking if a `yarn.lock` file exists. If the project is using Yarn, display all the instructions with Yarn commands and use Yarn to install packages in `init` and `eject` scripts. --- .travis.yml | 3 ++ packages/create-react-app/index.js | 52 +++++++++++++++++++------ packages/react-scripts/config/paths.js | 3 ++ packages/react-scripts/scripts/build.js | 17 ++++++-- packages/react-scripts/scripts/eject.js | 14 +++++-- packages/react-scripts/scripts/init.js | 45 +++++++++++++-------- packages/react-scripts/scripts/start.js | 6 ++- tasks/e2e.sh | 6 +++ 8 files changed, 110 insertions(+), 36 deletions(-) diff --git a/.travis.yml b/.travis.yml index 75383b87806..ec44c3294da 100644 --- a/.travis.yml +++ b/.travis.yml @@ -9,3 +9,6 @@ cache: - packages/create-react-app/node_modules - packages/react-scripts/node_modules script: tasks/e2e.sh +env: + - USE_YARN=no + - USE_YARN=yes diff --git a/packages/create-react-app/index.js b/packages/create-react-app/index.js index d6478a13545..23d8b5e8dea 100644 --- a/packages/create-react-app/index.js +++ b/packages/create-react-app/index.js @@ -101,26 +101,54 @@ function createApp(name, verbose, version) { process.chdir(root); console.log('Installing packages. This might take a couple minutes.'); - console.log('Installing react-scripts from npm...'); + console.log('Installing react-scripts...'); console.log(); run(root, appName, version, verbose, originalDirectory); } -function run(root, appName, version, verbose, originalDirectory) { - var installPackage = getInstallPackage(version); - var packageName = getPackageName(installPackage); +function install(packageToInstall, verbose, callback) { var args = [ - 'install', - verbose && '--verbose', - '--save-dev', - '--save-exact', - installPackage, - ].filter(function(e) { return e; }); - var proc = spawn('npm', args, {stdio: 'inherit'}); + 'add', + '--dev', + '--exact', + packageToInstall, + ]; + var proc = spawn('yarn', args, {stdio: 'inherit'}); + + var yarnExists = true; + proc.on('error', function (err) { + if (err.code === 'ENOENT') { + yarnExists = false; + } + }); proc.on('close', function (code) { + if (yarnExists) { + callback(code, 'yarn', args); + return; + } + // No Yarn installed, continuing with npm. + args = [ + 'install', + verbose && '--verbose', + '--save-dev', + '--save-exact', + packageToInstall, + ].filter(function(e) { return e; }); + var npmProc = spawn('npm', args, {stdio: 'inherit'}); + npmProc.on('close', function (code) { + callback(code, 'npm', args); + }); + }); +} + +function run(root, appName, version, verbose, originalDirectory) { + var packageToInstall = getInstallPackage(version); + var packageName = getPackageName(packageToInstall); + + install(packageToInstall, verbose, function (code, command, args) { if (code !== 0) { - console.error('`npm ' + args.join(' ') + '` failed'); + console.error('`' + command + ' ' + args.join(' ') + '` failed'); return; } diff --git a/packages/react-scripts/config/paths.js b/packages/react-scripts/config/paths.js index 1c154c36164..89cd2059cd7 100644 --- a/packages/react-scripts/config/paths.js +++ b/packages/react-scripts/config/paths.js @@ -43,6 +43,7 @@ module.exports = { appIndexJs: resolveApp('src/index.js'), appPackageJson: resolveApp('package.json'), appSrc: resolveApp('src'), + yarnLockFile: resolveApp('yarn.lock'), testsSetup: resolveApp('src/setupTests.js'), appNodeModules: resolveApp('node_modules'), ownNodeModules: resolveApp('node_modules'), @@ -62,6 +63,7 @@ module.exports = { appIndexJs: resolveApp('src/index.js'), appPackageJson: resolveApp('package.json'), appSrc: resolveApp('src'), + yarnLockFile: resolveApp('yarn.lock'), testsSetup: resolveApp('src/setupTests.js'), appNodeModules: resolveApp('node_modules'), // this is empty with npm3 but node resolution searches higher anyway: @@ -79,6 +81,7 @@ if (__dirname.indexOf(path.join('packages', 'react-scripts', 'config')) !== -1) appIndexJs: resolveOwn('../template/src/index.js'), appPackageJson: resolveOwn('../package.json'), appSrc: resolveOwn('../template/src'), + yarnLockFile: resolveOwn('../template/yarn.lock'), testsSetup: resolveOwn('../template/src/setupTests.js'), appNodeModules: resolveOwn('../node_modules'), ownNodeModules: resolveOwn('../node_modules'), diff --git a/packages/react-scripts/scripts/build.js b/packages/react-scripts/scripts/build.js index d0b92f6a73b..8b1cd4cc48f 100644 --- a/packages/react-scripts/scripts/build.js +++ b/packages/react-scripts/scripts/build.js @@ -21,6 +21,7 @@ require('dotenv').config({silent: true}); var chalk = require('chalk'); var fs = require('fs-extra'); var path = require('path'); +var pathExists = require('path-exists'); var filesize = require('filesize'); var gzipSize = require('gzip-size').sync; var rimrafSync = require('rimraf').sync; @@ -31,6 +32,8 @@ var checkRequiredFiles = require('react-dev-utils/checkRequiredFiles'); var recursive = require('recursive-readdir'); var stripAnsi = require('strip-ansi'); +var useYarn = pathExists.sync(paths.yarnLockFile); + // Warn and crash if required files are missing if (!checkRequiredFiles([paths.appHtml, paths.appIndexJs])) { process.exit(1); @@ -161,7 +164,11 @@ function build(previousSizeMap) { console.log('The ' + chalk.cyan('build') + ' folder is ready to be deployed.'); console.log('To publish it at ' + chalk.green(homepagePath) + ', run:'); console.log(); - console.log(' ' + chalk.cyan('npm') + ' install --save-dev gh-pages'); + if (useYarn) { + console.log(' ' + chalk.cyan('yarn') + ' add gh-pages'); + } else { + console.log(' ' + chalk.cyan('npm') + ' install --save-dev gh-pages'); + } console.log(); console.log('Add the following script in your ' + chalk.cyan('package.json') + '.'); console.log(); @@ -173,7 +180,7 @@ function build(previousSizeMap) { console.log(); console.log('Then run:'); console.log(); - console.log(' ' + chalk.cyan('npm') + ' run deploy'); + console.log(' ' + chalk.cyan(useYarn ? 'yarn' : 'npm') + ' run deploy'); console.log(); } else if (publicPath !== '/') { // "homepage": "http://mywebsite.com/project" @@ -200,7 +207,11 @@ function build(previousSizeMap) { console.log('The ' + chalk.cyan('build') + ' folder is ready to be deployed.'); console.log('You may also serve it locally with a static server:') console.log(); - console.log(' ' + chalk.cyan('npm') + ' install -g pushstate-server'); + if (useYarn) { + console.log(' ' + chalk.cyan('yarn') + ' global add pushstate-server'); + } else { + console.log(' ' + chalk.cyan('npm') + ' install -g pushstate-server'); + } console.log(' ' + chalk.cyan('pushstate-server') + ' build'); console.log(' ' + chalk.cyan(openCommand) + ' http://localhost:9000'); console.log(); diff --git a/packages/react-scripts/scripts/eject.js b/packages/react-scripts/scripts/eject.js index dbd4d64e4d5..7d4996665ec 100644 --- a/packages/react-scripts/scripts/eject.js +++ b/packages/react-scripts/scripts/eject.js @@ -10,6 +10,8 @@ var createJestConfig = require('../utils/createJestConfig'); var fs = require('fs'); var path = require('path'); +var pathExists = require('path-exists'); +var paths = require('../config/paths'); var prompt = require('react-dev-utils/prompt'); var rimrafSync = require('rimraf').sync; var spawnSync = require('cross-spawn').sync; @@ -143,9 +145,15 @@ prompt( ); console.log(); - console.log(cyan('Running npm install...')); - rimrafSync(ownPath); - spawnSync('npm', ['install'], {stdio: 'inherit'}); + if (pathExists.sync(paths.yarnLockFile)) { + console.log(cyan('Running yarn...')); + rimrafSync(ownPath); + spawnSync('yarn', [], {stdio: 'inherit'}); + } else { + console.log(cyan('Running npm install...')); + rimrafSync(ownPath); + spawnSync('npm', ['install'], {stdio: 'inherit'}); + } console.log(green('Ejected successfully!')); console.log(); diff --git a/packages/react-scripts/scripts/init.js b/packages/react-scripts/scripts/init.js index fa42f6dcee6..c9a4ea14ac4 100644 --- a/packages/react-scripts/scripts/init.js +++ b/packages/react-scripts/scripts/init.js @@ -17,6 +17,7 @@ module.exports = function(appPath, appName, verbose, originalDirectory) { var ownPackageName = require(path.join(__dirname, '..', 'package.json')).name; var ownPath = path.join(appPath, 'node_modules', ownPackageName); var appPackage = require(path.join(appPath, 'package.json')); + var useYarn = pathExists.sync(path.join(appPath, 'yarn.lock')); // Copy over some of the devDependencies appPackage.dependencies = appPackage.dependencies || {}; @@ -58,21 +59,31 @@ module.exports = function(appPath, appName, verbose, originalDirectory) { } }); - // Run another npm install for react and react-dom - console.log('Installing react and react-dom from npm...'); + // Run yarn or npm for react and react-dom + // TODO: having to do two npm/yarn installs is bad, can we avoid it? + var command; + var args; + + if (useYarn) { + command = 'yarn'; + args = ['add']; + } else { + command = 'npm'; + args = [ + 'install', + '--save', + verbose && '--verbose' + ].filter(function(e) { return e; }); + } + args.push('react', 'react-dom'); + + console.log('Installing react and react-dom using ' + command + '...'); console.log(); - // TODO: having to do two npm installs is bad, can we avoid it? - var args = [ - 'install', - 'react', - 'react-dom', - '--save', - verbose && '--verbose' - ].filter(function(e) { return e; }); - var proc = spawn('npm', args, {stdio: 'inherit'}); + + var proc = spawn(command, args, {stdio: 'inherit'}); proc.on('close', function (code) { if (code !== 0) { - console.error('`npm ' + args.join(' ') + '` failed'); + console.error('`' + command + ' ' + args.join(' ') + '` failed'); return; } @@ -91,23 +102,23 @@ module.exports = function(appPath, appName, verbose, originalDirectory) { console.log('Success! Created ' + appName + ' at ' + appPath); console.log('Inside that directory, you can run several commands:'); console.log(); - console.log(chalk.cyan(' npm start')); + console.log(chalk.cyan(' ' + command + ' start')); console.log(' Starts the development server.'); console.log(); - console.log(chalk.cyan(' npm run build')); + console.log(chalk.cyan(' ' + command + ' run build')); console.log(' Bundles the app into static files for production.'); console.log(); - console.log(chalk.cyan(' npm test')); + console.log(chalk.cyan(' ' + command + ' test')); console.log(' Starts the test runner.'); console.log(); - console.log(chalk.cyan(' npm run eject')); + console.log(chalk.cyan(' ' + command + ' run eject')); console.log(' Removes this tool and copies build dependencies, configuration files'); console.log(' and scripts into the app directory. If you do this, you can’t go back!'); console.log(); console.log('We suggest that you begin by typing:'); console.log(); console.log(chalk.cyan(' cd'), cdpath); - console.log(' ' + chalk.cyan('npm start')); + console.log(' ' + chalk.cyan(command + ' start')); if (readmeExists) { console.log(); console.log(chalk.yellow('You had a `README.md` file, we renamed it to `README.old.md`')); diff --git a/packages/react-scripts/scripts/start.js b/packages/react-scripts/scripts/start.js index 8a115dd8e27..5e996c71ddc 100644 --- a/packages/react-scripts/scripts/start.js +++ b/packages/react-scripts/scripts/start.js @@ -28,9 +28,13 @@ var checkRequiredFiles = require('react-dev-utils/checkRequiredFiles'); var formatWebpackMessages = require('react-dev-utils/formatWebpackMessages'); var openBrowser = require('react-dev-utils/openBrowser'); var prompt = require('react-dev-utils/prompt'); +var pathExists = require('path-exists'); var config = require('../config/webpack.config.dev'); var paths = require('../config/paths'); +var useYarn = pathExists.sync(paths.yarnLockFile); +var cli = useYarn ? 'yarn' : 'npm'; + // Warn and crash if required files are missing if (!checkRequiredFiles([paths.appHtml, paths.appIndexJs])) { process.exit(1); @@ -85,7 +89,7 @@ function setupCompiler(host, port, protocol) { console.log(' ' + chalk.cyan(protocol + '://' + host + ':' + port + '/')); console.log(); console.log('Note that the development build is not optimized.'); - console.log('To create a production build, use ' + chalk.cyan('npm run build') + '.'); + console.log('To create a production build, use ' + chalk.cyan(cli + ' run build') + '.'); console.log(); } diff --git a/tasks/e2e.sh b/tasks/e2e.sh index 88e1fdf4e20..094fba9e22b 100755 --- a/tasks/e2e.sh +++ b/tasks/e2e.sh @@ -53,6 +53,12 @@ set -x cd .. root_path=$PWD +if [ "$USE_YARN" = "yes" ] +then + # Install Yarn so that the test can use it to install packages. + npm install -g yarn +fi + npm install # Lint own code