[WEB] Un-ejected from create-react-app + Upgraded it to v1.0+ (webpack2, ...)

This commit is contained in:
Bruno Lemos
2017-05-25 01:37:29 -03:00
parent 2b870ca00e
commit 3c07f1902a
31 changed files with 2562 additions and 3685 deletions

View File

@@ -7,9 +7,12 @@ import './web/src/index.css'
import './web/src/reset.css'
import App from './src/'
import registerServiceWorker from './web/src/registerServiceWorker'
import { name as appName } from './package.json'
AppRegistry.registerComponent(appName, () => App)
AppRegistry.runApplication(appName, {
rootTag: document.getElementById('root'),
})
registerServiceWorker()

View File

@@ -1,7 +1,7 @@
import React from 'react'
import { withTheme } from 'styled-components/native'
import { TabNavigator } from 'react-navigation'
import { TabNavigator } from '../libs/navigation'
import Platform from '../libs/platform'
const _tabBarOptions = Platform.selectUsingRealOS({

View File

@@ -4,11 +4,11 @@ import React from 'react'
import styled, { ThemeProvider } from 'styled-components/native'
import { connect } from 'react-redux'
import { Platform, StatusBar } from 'react-native'
import { NavigationActions } from 'react-navigation'
import AppNavigatorContainer from './navigators/AppNavigatorContainer'
import FaviconBadge from '../libs/favicon-badge'
import { Helmet } from '../libs/helmet'
import { NavigationActions } from '../libs/navigation'
import { get } from '../utils/immutable'
import {

View File

@@ -1,2 +1,4 @@
// TODO: Make this a 100% React Component solution to work better on the web
export default from '../prompt'
import prompt from '../prompt'
export default prompt

View File

@@ -2,4 +2,3 @@
// TODO: Fix eslint for web imports
export * from 'react-helmet'
export default from 'react-helmet'

View File

@@ -1,7 +1,7 @@
/* eslint-disable react-native/no-color-literals */
// Source code from https://github.com/necolas/react-native-web/pull/411
import React, { Component, PropTypes } from 'react'
import React, { PureComponent, PropTypes } from 'react'
import { Animated, Dimensions, StyleSheet } from 'react-native'
const shortAnimTime = 200
@@ -17,7 +17,7 @@ const styles = StyleSheet.create({
},
})
class Modal extends Component {
export default class Modal extends PureComponent {
static displayName = 'Modal'
static propTypes = {
@@ -127,5 +127,3 @@ class Modal extends Component {
)
}
}
module.exports = Modal

View File

@@ -1,2 +0,0 @@
export * from 'react-navigation'
export default from 'react-navigation'

View File

@@ -1,3 +0,0 @@
export * from 'react-navigation'
export StackNavigator from 'react-navigation/lib/navigators/StackNavigator'
export TabNavigator from 'react-navigation/lib/navigators/TabNavigator'

View File

@@ -1,2 +1 @@
export * from 'react-native-prompt-android'
export default from 'react-native-prompt-android'

View File

@@ -1,13 +1,12 @@
// @flow
import { StackNavigator } from 'react-navigation'
import MainTabNavigator from './MainTabNavigator'
// import BrowserScreen from '../containers/screens/BrowserScreen';
import LoginScreen from '../containers/screens/LoginScreen'
import SplashScreen from '../containers/screens/SplashScreen'
import { StackNavigator } from '../libs/navigation'
export const routes = {
// browser: { screen: BrowserScreen },
login: { path: 'login', screen: LoginScreen },
main: { screen: MainTabNavigator },
splash: { screen: SplashScreen },

View File

@@ -1,9 +1,10 @@
// @flow
import { TabNavigator } from 'react-navigation'
import HomeScreen from '../containers/screens/HomeScreen'
import NotificationsScreen from '../containers/screens/NotificationsScreen'
import SettingsScreen from '../containers/screens/SettingsScreen'
import { TabNavigator } from '../libs/navigation'
import { tabBarOptions } from '../components/TabBar'
export const routes = {

View File

@@ -1,7 +1,5 @@
/* eslint-disable global-require */
if (process.env.NODE_ENV === 'production') {
module.exports = require('./configure.prod')
} else {
module.exports = require('./configure.dev')
}
export default (process.env.NODE_ENV === 'production'
? require('./configure.prod.js').default
: require('./configure.dev.js').default)

View File

@@ -1,7 +1,5 @@
/* eslint-disable global-require */
if (process.env.NODE_ENV === 'production') {
module.exports = require('./setup.prod')
} else {
module.exports = require('./setup.dev')
}
export default (process.env.NODE_ENV === 'production'
? require('./setup.prod').default
: require('./setup.dev').default)

18
web/.gitignore vendored
View File

@@ -1,15 +1,21 @@
# See http://help.github.com/ignore-files/ for more about ignoring files.
# See https://help.github.com/ignore-files/ for more about ignoring files.
# dependencies
/node_modules
node_modules
# testing
/coverage
coverage
# production
/build
build
# misc
.DS_Store
.env
npm-debug.log
.env.local
.env.development.local
.env.test.local
.env.production.local
npm-debug.log*
yarn-debug.log*
yarn-error.log*

View File

@@ -1,70 +0,0 @@
var ExtractTextPlugin = require('extract-text-webpack-plugin')
module.exports = {
BABEL_STAGE_0: {
toArray: 'presets',
getDev: function() {
return require.resolve('babel-preset-stage-0')
},
},
DECORATORS: {
toArray: 'babelPlugins',
getDev: function() {
return require.resolve('babel-plugin-transform-decorators-legacy')
},
},
SASS: {
toArray: 'loaders',
fileRegex: /\.(scss|sass)/,
getDev: function() {
return {
test: /(\.scss|\.sass)$/,
loader: 'style!css!postcss!sass',
}
},
getProd: function() {
return {
test: /(\.scss|\.sass)$/,
loader: ExtractTextPlugin.extract('style', 'css!postcss!sass'),
}
},
},
LESS: {
toArray: 'loaders',
fileRegex: /\.less$/,
getDev: function() {
return {
test: /\.less$/,
loader: 'style!css!postcss!less',
}
},
getProd: function() {
return {
test: /\.less/,
loader: ExtractTextPlugin.extract('style', 'css!postcss!less'),
}
},
},
STYLUS: {
toArray: 'loaders',
fileRegex: /\.styl$/,
getDev: function() {
return {
test: /\.styl/,
loader: 'style!css!postcss!stylus',
}
},
getProd: function() {
return {
test: /\.styl/,
loader: ExtractTextPlugin.extract('style', 'css!postcss!stylus'),
}
},
},
CSS_MODULES: {
config: {
dev: 'style!css?modules&camelCase&importLoaders=1&localIdentName=[name]__[local]___[hash:base64:5]!postcss',
prod: 'style!css?modules&camelCase&-autoprefixer&importLoaders=1!postcss',
},
},
}

View File

@@ -1,28 +0,0 @@
// Grab NODE_ENV and REACT_APP_* environment variables and prepare them to be
// injected into the application via DefinePlugin in Webpack configuration.
var REACT_APP = /^REACT_APP_/i
function getClientEnvironment(publicUrl) {
var processEnv = Object.keys(process.env)
.filter(key => REACT_APP.test(key))
.reduce(
(env, key) => {
env[key] = JSON.stringify(process.env[key])
return env
},
{
// Useful for determining whether were running in production mode.
// Most importantly, it switches React into the correct mode.
NODE_ENV: JSON.stringify(process.env.NODE_ENV || 'development'),
// Useful for resolving the correct path to static assets in `public`.
// For example, <img src={process.env.PUBLIC_URL + '/img/logo.png'} />.
// This should only be used as an escape hatch. Normally you would put
// images into the `src` and `import` them in code to get their paths.
PUBLIC_URL: JSON.stringify(publicUrl),
},
)
return { 'process.env': processEnv }
}
module.exports = getClientEnvironment

View File

@@ -1,40 +0,0 @@
var customizers = require('./customizers')
function getCustomConfig(_prod) {
var prod = _prod || false
var env = env || {} // eslint-disable-line
var result = Object.keys(customizers).reduce(
function(finalConfig, customizerKey) {
var customizer = customizers[customizerKey]
if (customizer.prod === false && prod === true) {
return finalConfig
}
var envValue = process.env['REACT_APP_' + customizerKey]
if (env && envValue && envValue !== 'false') {
if (customizer.toArray) {
var getCustomizer =
(prod ? customizer.getProd : customizer.getDev) || customizer.getDev
finalConfig[customizer.toArray].push(getCustomizer())
}
if (customizer.fileRegex) {
finalConfig.excludedFilesRegex.push(customizer.fileRegex)
}
finalConfig.values[customizerKey] = customizer.config || true
}
return finalConfig
},
{
presets: [],
babelPlugins: [],
plugins: [],
loaders: [],
values: {},
excludedFilesRegex: [],
},
)
return result
}
module.exports = getCustomConfig

View File

@@ -1,12 +0,0 @@
// This is a custom Jest transformer turning style imports into empty objects.
// http://facebook.github.io/jest/docs/tutorial-webpack.html
module.exports = {
process() {
return 'module.exports = {};'
},
getCacheKey(fileData, filename) {
// The output is always the same.
return 'cssTransform'
},
}

View File

@@ -1,10 +0,0 @@
const path = require('path')
// This is a custom Jest transformer turning file imports into filenames.
// http://facebook.github.io/jest/docs/tutorial-webpack.html
module.exports = {
process(src, filename) {
return 'module.exports = ' + JSON.stringify(path.basename(filename)) + ';'
},
}

View File

@@ -1,54 +0,0 @@
var path = require('path')
var fs = require('fs')
// Make sure any symlinks in the project folder are resolved:
// https://github.com/facebookincubator/create-react-app/issues/637
var appDirectory = fs.realpathSync(process.cwd())
function resolveApp(relativePath) {
return path.resolve(
appDirectory,
process.env.WEB_ROOT_PATH || './',
relativePath,
)
}
// We support resolving modules according to `NODE_PATH`.
// This lets you use absolute paths in imports inside large monorepos:
// https://github.com/facebookincubator/create-react-app/issues/253.
// It works similar to `NODE_PATH` in Node itself:
// https://nodejs.org/api/modules.html#modules_loading_from_the_global_folders
// We will export `nodePaths` as an array of absolute paths.
// It will then be used by Webpack configs.
// Jest doesnt need this because it already handles `NODE_PATH` out of the box.
// Note that unlike in Node, only *relative* paths from `NODE_PATH` are honored.
// Otherwise, we risk importing Node.js core modules into an app instead of Webpack shims.
// https://github.com/facebookincubator/create-react-app/issues/1023#issuecomment-265344421
var nodePaths = (process.env.NODE_PATH || '')
.split(process.platform === 'win32' ? ';' : ':')
.filter(Boolean)
.filter(folder => !path.isAbsolute(folder))
.map(resolveApp)
// config after eject: we're in ./config/
module.exports = {
appBuild: resolveApp('build'),
appPublic: resolveApp('public'),
appHtml: resolveApp('public/index.html'),
appIndexJs: resolveApp('../index.web.js'),
appPackageJson: resolveApp('package.json'),
appSrc: [
resolveApp('../src'),
resolveApp('../index.web.js'),
resolveApp('../node_modules/react-native-vector-icons'),
resolveApp('../node_modules/react-native-tab-view/src'),
],
yarnLockFile: resolveApp('yarn.lock'),
testsSetup: resolveApp('src/setupTests.js'),
appNodeModules: resolveApp('node_modules'),
ownNodeModules: resolveApp('node_modules'),
nodePaths: [resolveApp('node_modules'), ...nodePaths],
}

View File

@@ -1,14 +0,0 @@
if (typeof Promise === 'undefined') {
// Rejection tracking prevents a common issue where React gets into an
// inconsistent state due to an error, but it gets swallowed by a Promise,
// and the user has no idea what causes React's erratic future behavior.
require('promise/lib/rejection-tracking').enable()
window.Promise = require('promise/lib/es6-extensions.js')
}
// fetch() polyfill for making API calls.
require('whatwg-fetch')
// Object.assign() is commonly used with React.
// It will use the native implementation if it's present and isn't buggy.
Object.assign = require('object-assign')

View File

@@ -1,219 +0,0 @@
var autoprefixer = require('autoprefixer')
var webpack = require('webpack')
var HtmlWebpackPlugin = require('html-webpack-plugin')
var CaseSensitivePathsPlugin = require('case-sensitive-paths-webpack-plugin')
var InterpolateHtmlPlugin = require('react-dev-utils/InterpolateHtmlPlugin')
var WatchMissingNodeModulesPlugin = require('react-dev-utils/WatchMissingNodeModulesPlugin')
var getClientEnvironment = require('./env')
var paths = require('./paths')
var getCustomConfig = require('./get-custom-config')
// Webpack uses `publicPath` to determine where the app is being served from.
// In development, we always serve from the root. This makes config easier.
var publicPath = '/'
// `publicUrl` is just like `publicPath`, but we will provide it to our app
// as %PUBLIC_URL% in `index.html` and `process.env.PUBLIC_URL` in JavaScript.
// Omit trailing slash as %PUBLIC_PATH%/xyz looks better than %PUBLIC_PATH%xyz.
var publicUrl = ''
// Get environment variables to inject into our app.
var env = getClientEnvironment(publicUrl)
//Get custom configuration for injecting plugins, presets and loaders
var customConfig = getCustomConfig(false)
// This is the development configuration.
// It is focused on developer experience and fast rebuilds.
// The production configuration is different and lives in a separate file.
module.exports = {
// You may want 'eval' instead if you prefer to see the compiled output in DevTools.
// See the discussion in https://github.com/facebookincubator/create-react-app/issues/343.
devtool: 'cheap-module-source-map',
// These are the "entry points" to our application.
// This means they will be the "root" imports that are included in JS bundle.
// The first two entry points enable "hot" CSS and auto-refreshes for JS.
entry: [
// Include an alternative client for WebpackDevServer. A client's job is to
// connect to WebpackDevServer by a socket and get notified about changes.
// When you save a file, the client will either apply hot updates (in case
// of CSS changes), or refresh the page (in case of JS changes). When you
// make a syntax error, this client will display a syntax error overlay.
// Note: instead of the default WebpackDevServer client, we use a custom one
// to bring better experience for Create React App users. You can replace
// the line below with these two lines if you prefer the stock client:
// require.resolve('webpack-dev-server/client') + '?/',
// require.resolve('webpack/hot/dev-server'),
require.resolve('react-dev-utils/webpackHotDevClient'),
// We ship a few polyfills by default:
require.resolve('./polyfills'),
// Finally, this is your app's code:
paths.appIndexJs,
// We include the app code last so that if there is a runtime error during
// initialization, it doesn't blow up the WebpackDevServer client, and
// changing JS code would still trigger a refresh.
],
output: {
// Next line is not used in dev but WebpackDevServer crashes without it:
path: paths.appBuild,
// Add /* filename */ comments to generated require()s in the output.
pathinfo: true,
// This does not produce a real file. It's just the virtual path that is
// served by WebpackDevServer in development. This is the JS bundle
// containing code from all our entry points, and the Webpack runtime.
filename: 'static/js/bundle.js',
// This is the URL that app is served from. We use "/" in development.
publicPath: publicPath,
},
resolve: {
// This allows you to set a fallback for where Webpack should look for modules.
// We read `NODE_PATH` environment variable in `paths.js` and pass paths here.
// We use `fallback` instead of `root` because we want `node_modules` to "win"
// if there any conflicts. This matches Node resolution mechanism.
// https://github.com/facebookincubator/create-react-app/issues/253
fallback: paths.nodePaths,
// These are the reasonable defaults supported by the Node ecosystem.
// We also include JSX as a common component filename extension to support
// some tools, although we do not recommend using it, see:
// https://github.com/facebookincubator/create-react-app/issues/290
extensions: ['.web.js', '.js', '.json', '.jsx', ''],
alias: {
// Support React Native Web
// https://www.smashingmagazine.com/2016/08/a-glimpse-into-the-future-with-react-native-for-web/
'react-native': 'react-native-web',
},
},
resolveLoader: {
root: [paths.ownNodeModules, paths.appNodeModules],
moduleTemplates: ['*-loader'],
},
module: {
// First, run the linter.
// It's important to do this before Babel processes the JS.
// TODO: Reenable linter
// preLoaders: [
// {
// test: /\.(js|jsx)$/,
// loader: 'eslint',
// include: paths.appSrc,
// }
// ],
loaders: [
// Default loader: load all assets that are not handled
// by other loaders with the url loader.
// Note: This list needs to be updated with every change of extensions
// the other loaders match.
// E.g., when adding a loader for a new supported file extension,
// we need to add the supported extension to this loader too.
// Add one new line in `exclude` for each loader.
//
// "file" loader makes sure those assets get served by WebpackDevServer.
// When you `import` an asset, you get its (virtual) filename.
// In production, they would get copied to the `build` folder.
// "url" loader works like "file" loader except that it embeds assets
// smaller than specified limit in bytes as data URLs to avoid requests.
// A missing `test` is equivalent to a match.
{
exclude: [
/\.html$/,
/\.(js|jsx)$/,
/\.css$/,
/\.json$/,
/\.svg$/,
].concat(customConfig.excludedFilesRegex),
loader: 'url',
query: {
limit: 10000,
name: 'static/media/[name].[hash:8].[ext]',
},
},
// Process JS with Babel.
{
test: /\.(js|jsx)$/,
include: paths.appSrc,
loader: 'babel',
query: {
// This is a feature of `babel-loader` for webpack (not Babel itself).
// It enables caching results in ./node_modules/.cache/babel-loader/
// directory for faster rebuilds.
cacheDirectory: true,
},
},
// "postcss" loader applies autoprefixer to our CSS.
// "css" loader resolves paths in CSS and adds assets as dependencies.
// "style" loader turns CSS into JS modules that inject <style> tags.
// In production, we use a plugin to extract that CSS to a file, but
// in development "style" loader enables hot editing of CSS.
{
test: /\.css$/,
loader: customConfig.values.CSS_MODULES
? customConfig.values.CSS_MODULES.dev
: 'style!css?importLoaders=1!postcss',
},
// JSON is not enabled by default in Webpack but both Node and Browserify
// allow it implicitly so we also enable it.
{
test: /\.json$/,
loader: 'json',
},
// "file" loader for svg
{
test: /\.svg$/,
loader: 'file',
query: {
name: 'static/media/[name].[hash:8].[ext]',
},
},
].concat(customConfig.loaders),
},
// We use PostCSS for autoprefixing only.
postcss: function() {
return [
autoprefixer({
browsers: [
'>1%',
'last 4 versions',
'Firefox ESR',
'not ie < 9', // React doesn't support IE8 anyway
],
}),
]
},
plugins: [
// Makes the public URL available as %PUBLIC_URL% in index.html, e.g.:
// <link rel="shortcut icon" href="%PUBLIC_URL%/favicon.ico">
// In development, this will be an empty string.
new InterpolateHtmlPlugin({
PUBLIC_URL: publicUrl,
}),
// Generates an `index.html` file with the <script> injected.
new HtmlWebpackPlugin({
inject: true,
template: paths.appHtml,
}),
// Makes some environment variables available to the JS code, for example:
// if (process.env.NODE_ENV === 'development') { ... }. See `./env.js`.
new webpack.DefinePlugin(
Object.assign({}, env, {
__DEV__: true,
}),
),
// This is necessary to emit hot updates (currently CSS only):
new webpack.HotModuleReplacementPlugin(),
// Watcher doesn't work well if you mistype casing in a path so we use
// a plugin that prints an error when you attempt to do this.
// See https://github.com/facebookincubator/create-react-app/issues/240
new CaseSensitivePathsPlugin(),
// If you require a missing module and then `npm install` it, you still have
// to restart the development server for Webpack to discover it. This plugin
// makes the discovery automatic so you don't have to restart.
// See https://github.com/facebookincubator/create-react-app/issues/186
new WatchMissingNodeModulesPlugin(paths.appNodeModules),
].concat(customConfig.plugins),
// Some libraries import Node modules but don't use them in the browser.
// Tell Webpack to provide empty mocks for them so importing them works.
node: {
fs: 'empty',
net: 'empty',
tls: 'empty',
},
}

View File

@@ -1,262 +0,0 @@
var autoprefixer = require('autoprefixer')
var webpack = require('webpack')
var HtmlWebpackPlugin = require('html-webpack-plugin')
var ExtractTextPlugin = require('extract-text-webpack-plugin')
var ManifestPlugin = require('webpack-manifest-plugin')
var InterpolateHtmlPlugin = require('react-dev-utils/InterpolateHtmlPlugin')
var url = require('url')
var BundleAnalyzerPlugin = require('webpack-bundle-analyzer')
.BundleAnalyzerPlugin
var paths = require('./paths')
var getClientEnvironment = require('./env')
var getCustomConfig = require('./get-custom-config')
function ensureSlash(path, needsSlash) {
var hasSlash = path.endsWith('/')
if (hasSlash && !needsSlash) {
return path.substr(path, path.length - 1)
} else if (!hasSlash && needsSlash) {
return path + '/'
} else {
return path
}
}
// We use "homepage" field to infer "public path" at which the app is served.
// Webpack needs to know it to put the right <script> hrefs into HTML even in
// single-page apps that may serve index.html for nested URLs like /todos/42.
// We can't use a relative path in HTML because we don't want to load something
// like /todos/42/static/js/bundle.7289d.js. We have to know the root.
var homepagePath = require(paths.appPackageJson).homepage
var homepagePathname = homepagePath ? url.parse(homepagePath).pathname : '/'
// Webpack uses `publicPath` to determine where the app is being served from.
// It requires a trailing slash, or the file assets will get an incorrect path.
var publicPath = ensureSlash(homepagePathname, true)
// `publicUrl` is just like `publicPath`, but we will provide it to our app
// as %PUBLIC_URL% in `index.html` and `process.env.PUBLIC_URL` in JavaScript.
// Omit trailing slash as %PUBLIC_PATH%/xyz looks better than %PUBLIC_PATH%xyz.
var publicUrl = ensureSlash(homepagePathname, false)
// Get environment variables to inject into our app.
var env = getClientEnvironment(publicUrl)
//Get custom configuration for injecting plugins, presets and loaders
var customConfig = getCustomConfig(true)
// Assert this just to be safe.
// Development builds of React are slow and not intended for production.
if (env['process.env'].NODE_ENV !== '"production"') {
throw new Error('Production builds must have NODE_ENV=production.')
}
// This is the production configuration.
// It compiles slowly and is focused on producing a fast and minimal bundle.
// The development configuration is different and lives in a separate file.
module.exports = {
// Don't attempt to continue if there are any errors.
bail: true,
// We generate sourcemaps in production. This is slow but gives good results.
// You can exclude the *.map files from the build during deployment.
devtool: 'source-map',
// In production, we only want to load the polyfills and the app code.
entry: [require.resolve('./polyfills'), paths.appIndexJs],
output: {
// The build folder.
path: paths.appBuild,
// Generated JS file names (with nested folders).
// There will be one main bundle, and one file per asynchronous chunk.
// We don't currently advertise code splitting but Webpack supports it.
filename: 'static/js/[name].[chunkhash:8].js',
chunkFilename: 'static/js/[name].[chunkhash:8].chunk.js',
// We inferred the "public path" (such as / or /my-project) from homepage.
publicPath: publicPath,
},
resolve: {
// This allows you to set a fallback for where Webpack should look for modules.
// We read `NODE_PATH` environment variable in `paths.js` and pass paths here.
// We use `fallback` instead of `root` because we want `node_modules` to "win"
// if there any conflicts. This matches Node resolution mechanism.
// https://github.com/facebookincubator/create-react-app/issues/253
fallback: paths.nodePaths,
// These are the reasonable defaults supported by the Node ecosystem.
// We also include JSX as a common component filename extension to support
// some tools, although we do not recommend using it, see:
// https://github.com/facebookincubator/create-react-app/issues/290
extensions: ['.web.js', '.js', '.json', '.jsx', ''],
alias: {
// Support React Native Web
// https://www.smashingmagazine.com/2016/08/a-glimpse-into-the-future-with-react-native-for-web/
'react-native': 'react-native-web',
},
},
resolveLoader: {
root: [paths.ownNodeModules, paths.appNodeModules],
moduleTemplates: ['*-loader'],
},
module: {
// First, run the linter.
// It's important to do this before Babel processes the JS.
// preLoaders: [
// {
// test: /\.(js|jsx)$/,
// loader: 'eslint',
// include: paths.appSrc
// }
// ],
loaders: [
// Default loader: load all assets that are not handled
// by other loaders with the url loader.
// Note: This list needs to be updated with every change of extensions
// the other loaders match.
// E.g., when adding a loader for a new supported file extension,
// we need to add the supported extension to this loader too.
// Add one new line in `exclude` for each loader.
//
// "file" loader makes sure those assets end up in the `build` folder.
// When you `import` an asset, you get its filename.
// "url" loader works just like "file" loader but it also embeds
// assets smaller than specified size as data URLs to avoid requests.
{
exclude: [
/\.html$/,
/\.(js|jsx)$/,
/\.css$/,
/\.json$/,
/\.svg$/,
].concat(customConfig.excludedFilesRegex),
loader: 'url',
query: {
limit: 10000,
name: 'static/media/[name].[hash:8].[ext]',
},
},
// Process JS with Babel.
{
test: /\.(js|jsx)$/,
include: paths.appSrc,
loader: 'babel',
},
// The notation here is somewhat confusing.
// "postcss" loader applies autoprefixer to our CSS.
// "css" loader resolves paths in CSS and adds assets as dependencies.
// "style" loader normally turns CSS into JS modules injecting <style>,
// but unlike in development configuration, we do something different.
// `ExtractTextPlugin` first applies the "postcss" and "css" loaders
// (second argument), then grabs the result CSS and puts it into a
// separate file in our build process. This way we actually ship
// a single CSS file in production instead of JS code injecting <style>
// tags. If you use code splitting, however, any async bundles will still
// use the "style" loader inside the async code so CSS from them won't be
// in the main CSS file.
{
test: /\.css$/,
loader: ExtractTextPlugin.extract.apply(
null,
customConfig.values.CSS_MODULES
? ['style', 'css?modules&importLoaders=1', 'postcss']
: ['style', 'css?importLoaders=1', 'postcss'],
),
// Note: this won't work without `new ExtractTextPlugin()` in `plugins`.
},
// JSON is not enabled by default in Webpack but both Node and Browserify
// allow it implicitly so we also enable it.
{
test: /\.json$/,
loader: 'json',
},
// "file" loader for svg
{
test: /\.svg$/,
loader: 'file',
query: {
name: 'static/media/[name].[hash:8].[ext]',
},
},
].concat(customConfig.loaders),
},
// We use PostCSS for autoprefixing only.
postcss: function() {
return [
autoprefixer({
browsers: [
'>1%',
'last 4 versions',
'Firefox ESR',
'not ie < 9', // React doesn't support IE8 anyway
],
}),
]
},
plugins: [
// Makes the public URL available as %PUBLIC_URL% in index.html, e.g.:
// <link rel="shortcut icon" href="%PUBLIC_URL%/favicon.ico">
// In production, it will be an empty string unless you specify "homepage"
// in `package.json`, in which case it will be the pathname of that URL.
new InterpolateHtmlPlugin({
PUBLIC_URL: publicUrl,
}),
// Generates an `index.html` file with the <script> injected.
new HtmlWebpackPlugin({
inject: true,
template: paths.appHtml,
minify: {
removeComments: true,
collapseWhitespace: true,
removeRedundantAttributes: true,
useShortDoctype: true,
removeEmptyAttributes: true,
removeStyleLinkTypeAttributes: true,
keepClosingSlash: true,
minifyJS: true,
minifyCSS: true,
minifyURLs: true,
},
}),
// Makes some environment variables available to the JS code, for example:
// if (process.env.NODE_ENV === 'production') { ... }. See `./env.js`.
// It is absolutely essential that NODE_ENV was set to production here.
// Otherwise React will be compiled in the very slow development mode.
new webpack.DefinePlugin(
Object.assign({}, env, {
__DEV__: false,
}),
),
// This helps ensure the builds are consistent if source hasn't changed:
new webpack.optimize.OccurrenceOrderPlugin(),
// Try to dedupe duplicated modules, if any:
new webpack.optimize.DedupePlugin(),
// Minify the code.
new webpack.optimize.UglifyJsPlugin({
compress: {
screw_ie8: true, // React doesn't support IE8
warnings: false,
},
mangle: {
screw_ie8: true,
},
output: {
comments: false,
screw_ie8: true,
},
}),
// Note: this won't work without ExtractTextPlugin.extract(..) in `loaders`.
new ExtractTextPlugin('static/css/[name].[contenthash:8].css'),
// Generate a manifest file which contains a mapping of all asset filenames
// to their corresponding output file so that tools can pick it up without
// having to parse `index.html`.
new ManifestPlugin({
fileName: 'asset-manifest.json',
}),
new BundleAnalyzerPlugin({
analyzerMode: 'static',
openAnalyzer: false,
}),
].concat(customConfig.plugins),
// Some libraries import Node modules but don't use them in the browser.
// Tell Webpack to provide empty mocks for them so importing them works.
node: {
fs: 'empty',
net: 'empty',
tls: 'empty',
},
}

49
web/cra.config.js Normal file
View File

@@ -0,0 +1,49 @@
module.exports = {
paths: (paths, { resolvePath }) => {
const sourcesToCompile = paths.sourcesToCompile.concat([
resolvePath('../src'),
resolvePath('../index.web.js'),
resolvePath('../node_modules/react-native-vector-icons'),
resolvePath('../node_modules/react-native-tab-view/src'),
])
return Object.assign({}, paths, { sourcesToCompile })
},
webpack: (oldConfig, { isDevelopment }) => {
const babelLoader = require.resolve('babel-loader')
const ModuleScopePlugin = require('react-dev-utils/ModuleScopePlugin')
const webpack = require('webpack')
return Object.assign({}, oldConfig, {
module: Object.assign({}, oldConfig.module, {
rules: oldConfig.module.rules.map(rule => {
if (rule.loader === babelLoader) {
return Object.assign({}, rule, {
options: Object.assign({}, rule.options, {
plugins: (rule.options.plugins || [])
.concat([
require.resolve('babel-plugin-transform-decorators-legacy'),
]),
}),
})
}
return rule
}),
}),
plugins: oldConfig.plugins.concat([
new webpack.DefinePlugin({ __DEV__: isDevelopment }),
]),
resolve: Object.assign({}, oldConfig.resolve, {
alias: Object.assign({}, oldConfig.resolve.alias, {
'react-navigation': 'react-navigation/lib/react-navigation.js',
}),
extensions: ['.web.js'].concat(oldConfig.resolve.extensions),
plugins: oldConfig.resolve.plugins.filter(
plugin => !(plugin instanceof ModuleScopePlugin),
),
}),
})
},
}

View File

@@ -3,79 +3,29 @@
"version": "0.8.1",
"private": true,
"homepage": "https://brunolemos.github.io/devhub/",
"devDependencies": {
"autoprefixer": "6.5.1",
"babel-core": "6.17.0",
"babel-eslint": "7.1.1",
"babel-jest": "18.0.0",
"babel-loader": "6.2.10",
"babel-plugin-transform-decorators-legacy": "^1.3.4",
"babel-preset-react-app": "^2.0.1",
"babel-preset-stage-0": "^6.5.0",
"babel-runtime": "^6.20.0",
"case-sensitive-paths-webpack-plugin": "1.1.4",
"chalk": "1.1.3",
"connect-history-api-fallback": "1.3.0",
"cross-spawn": "4.0.2",
"css-loader": "0.26.0",
"detect-port": "1.0.1",
"dotenv": "2.0.0",
"eslint": "^3.15.0",
"eslint-config-airbnb": "^14.1.0",
"eslint-config-react-app": "^0.5.0",
"eslint-loader": "1.6.0",
"eslint-plugin-flowtype": "2.21.0",
"eslint-plugin-import": "^2.2.0",
"eslint-plugin-jsx-a11y": "^3.0.2 || ^4.0.0",
"eslint-plugin-react": "^6.9.0",
"extract-text-webpack-plugin": "1.0.1",
"file-loader": "0.9.0",
"filesize": "3.3.0",
"fs-extra": "0.30.0",
"gh-pages": "^1.0.0",
"gzip-size": "3.0.0",
"html-webpack-plugin": "2.24.0",
"http-proxy-middleware": "0.17.2",
"jest": "18.1.0",
"json-loader": "0.5.4",
"less": "^2.7.1",
"less-loader": "^2.2.3",
"node-babel": "^0.1.2",
"node-sass": "^3.10.0",
"object-assign": "4.1.0",
"postcss-loader": "1.0.0",
"promise": "7.1.1",
"pushstate-server": "^3.0.0",
"react-dev-utils": "^0.4.2",
"recursive-readdir": "2.1.0",
"rimraf": "2.5.4",
"sass-loader": "^4.0.2",
"strip-ansi": "3.0.1",
"style-loader": "0.13.1",
"stylus": "^0.54.5",
"stylus-loader": "^2.3.1",
"sw-precache": "^5.1.1",
"url-loader": "0.5.7",
"webpack": "1.14.0",
"webpack-bundle-analyzer": "^2.4.0",
"webpack-dev-server": "1.16.2",
"webpack-manifest-plugin": "1.1.0",
"whatwg-fetch": "1.0.0"
},
"dependencies": {
"bugsnag-js": "^3.2.0",
"react": "^15.5.4",
"react-dom": "^15.5.4",
"react-helmet": "^5.0.3",
"react-helmet": "^5.1.3",
"react-native-web": "^0.0.94"
},
"devDependencies": {
"babel-plugin-transform-decorators-legacy": "^1.3.4",
"extensible-react-scripts": "^1.0.6-4",
"gh-pages": "^1.0.0",
"react-dev-utils": "^2.0.1",
"serve": "^5.1.5",
"sw-precache": "^5.1.1"
},
"scripts": {
"build": "NODE_ENV=production babel-node scripts/build.js && sw-precache --config=sw-precache-config.js",
"build": "NODE_ENV=production react-scripts build && sw-precache --config=sw-precache-config.js",
"deploy": "gh-pages -d build",
"eject": "react-scripts eject",
"predeploy": "npm run build",
"serve": "pushstate-server build",
"start": "NODE_ENV=development babel-node scripts/start.js",
"test": "NODE_ENV=test babel-node scripts/test.js --env=jsdom"
"serve": "serve build",
"start": "react-scripts start",
"test": "react-scripts test --env=jsdom"
},
"jest": {
"collectCoverageFrom": [

View File

@@ -1,289 +0,0 @@
// Do this as the first thing so that any code reading it knows the right env.
process.env.NODE_ENV = 'production'
// Load environment variables from .env file. Suppress warnings using silent
// if this file is missing. dotenv will never modify any environment variables
// that have already been set.
// https://github.com/motdotla/dotenv
require('dotenv').config({ silent: true })
var chalk = require('chalk')
var fs = require('fs-extra')
var path = require('path')
var filesize = require('filesize')
var gzipSize = require('gzip-size').sync
var webpack = require('webpack')
var config = require('../config/webpack.config.prod')
var paths = require('../config/paths')
var checkRequiredFiles = require('react-dev-utils/checkRequiredFiles')
var recursive = require('recursive-readdir')
var stripAnsi = require('strip-ansi')
var useYarn = fs.existsSync(paths.yarnLockFile)
// Warn and crash if required files are missing
if (!checkRequiredFiles([paths.appHtml, paths.appIndexJs])) {
process.exit(1)
}
// Input: /User/dan/app/build/static/js/main.82be8.js
// Output: /static/js/main.js
function removeFileNameHash(fileName) {
return fileName
.replace(paths.appBuild, '')
.replace(/\/?(.*)(\.\w+)(\.js|\.css)/, (match, p1, p2, p3) => p1 + p3)
}
// Input: 1024, 2048
// Output: "(+1 KB)"
function getDifferenceLabel(currentSize, previousSize) {
var FIFTY_KILOBYTES = 1024 * 50
var difference = currentSize - previousSize
var fileSize = !Number.isNaN(difference) ? filesize(difference) : 0
if (difference >= FIFTY_KILOBYTES) {
return chalk.red('+' + fileSize)
} else if (difference < FIFTY_KILOBYTES && difference > 0) {
return chalk.yellow('+' + fileSize)
} else if (difference < 0) {
return chalk.green(fileSize)
} else {
return ''
}
}
// First, read the current file sizes in build directory.
// This lets us display how much they changed later.
recursive(paths.appBuild, (err, fileNames) => {
var previousSizeMap = (fileNames || [])
.filter(fileName => /\.(js|css)$/.test(fileName))
.reduce((memo, fileName) => {
var contents = fs.readFileSync(fileName)
var key = removeFileNameHash(fileName)
memo[key] = gzipSize(contents)
return memo
}, {})
// Remove all content but keep the directory so that
// if you're in it, you don't end up in Trash
fs.emptyDirSync(paths.appBuild)
// Start the webpack build
build(previousSizeMap)
// Merge with the public folder
copyPublicFolder()
})
// Print a detailed summary of build files.
function printFileSizes(stats, previousSizeMap) {
var assets = stats
.toJson()
.assets.filter(asset => /\.(js|css)$/.test(asset.name))
.map(asset => {
var fileContents = fs.readFileSync(paths.appBuild + '/' + asset.name)
var size = gzipSize(fileContents)
var previousSize = previousSizeMap[removeFileNameHash(asset.name)]
var difference = getDifferenceLabel(size, previousSize)
return {
folder: path.join('build', path.dirname(asset.name)),
name: path.basename(asset.name),
size: size,
sizeLabel: filesize(size) + (difference ? ' (' + difference + ')' : ''),
}
})
assets.sort((a, b) => b.size - a.size)
var longestSizeLabelLength = Math.max.apply(
null,
assets.map(a => stripAnsi(a.sizeLabel).length),
)
assets.forEach(asset => {
var sizeLabel = asset.sizeLabel
var sizeLength = stripAnsi(sizeLabel).length
if (sizeLength < longestSizeLabelLength) {
var rightPadding = ' '.repeat(longestSizeLabelLength - sizeLength)
sizeLabel += rightPadding
}
console.log(
' ' +
sizeLabel +
' ' +
chalk.dim(asset.folder + path.sep) +
chalk.cyan(asset.name),
)
})
}
// Print out errors
function printErrors(summary, errors) {
console.log(chalk.red(summary))
console.log()
errors.forEach(err => {
console.log(err.message || err)
console.log()
})
}
// Create the production build and print the deployment instructions.
function build(previousSizeMap) {
console.log('Creating an optimized production build...')
webpack(config).run((err, stats) => {
if (err) {
printErrors('Failed to compile.', [err])
process.exit(1)
}
if (stats.compilation.errors.length) {
printErrors('Failed to compile.', stats.compilation.errors)
process.exit(1)
}
if (process.env.CI && stats.compilation.warnings.length) {
printErrors('Failed to compile.', stats.compilation.warnings)
process.exit(1)
}
console.log(chalk.green('Compiled successfully.'))
console.log()
console.log('File sizes after gzip:')
console.log()
printFileSizes(stats, previousSizeMap)
console.log()
var openCommand = process.platform === 'win32' ? 'start' : 'open'
var appPackage = require(paths.appPackageJson)
var homepagePath = appPackage.homepage
var publicPath = config.output.publicPath
if (homepagePath && homepagePath.indexOf('.github.io/') !== -1) {
// "homepage": "http://user.github.io/project"
console.log(
'The project was built assuming it is hosted at ' +
chalk.green(publicPath) +
'.',
)
console.log(
'You can control this with the ' +
chalk.green('homepage') +
' field in your ' +
chalk.cyan('package.json') +
'.',
)
console.log()
console.log(
'The ' + chalk.cyan('build') + ' folder is ready to be deployed.',
)
console.log('To publish it at ' + chalk.green(homepagePath) + ', run:')
// If script deploy has been added to package.json, skip the instructions
if (typeof appPackage.scripts.deploy === 'undefined') {
console.log()
if (useYarn) {
console.log(' ' + chalk.cyan('yarn') + ' add --dev 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()
console.log(' ' + chalk.dim('// ...'))
console.log(' ' + chalk.yellow('"scripts"') + ': {')
console.log(' ' + chalk.dim('// ...'))
console.log(
' ' +
chalk.yellow('"predeploy"') +
': ' +
chalk.yellow('"npm run build",'),
)
console.log(
' ' +
chalk.yellow('"deploy"') +
': ' +
chalk.yellow('"gh-pages -d build"'),
)
console.log(' }')
console.log()
console.log('Then run:')
}
console.log()
console.log(' ' + chalk.cyan(useYarn ? 'yarn' : 'npm') + ' run deploy')
console.log()
} else if (publicPath !== '/') {
// "homepage": "http://mywebsite.com/project"
console.log(
'The project was built assuming it is hosted at ' +
chalk.green(publicPath) +
'.',
)
console.log(
'You can control this with the ' +
chalk.green('homepage') +
' field in your ' +
chalk.cyan('package.json') +
'.',
)
console.log()
console.log(
'The ' + chalk.cyan('build') + ' folder is ready to be deployed.',
)
console.log()
} else {
// no homepage or "homepage": "http://mywebsite.com"
console.log(
'The project was built assuming it is hosted at the server root.',
)
if (homepagePath) {
// "homepage": "http://mywebsite.com"
console.log(
'You can control this with the ' +
chalk.green('homepage') +
' field in your ' +
chalk.cyan('package.json') +
'.',
)
console.log()
} else {
// no homepage
console.log(
'To override this, specify the ' +
chalk.green('homepage') +
' in your ' +
chalk.cyan('package.json') +
'.',
)
console.log('For example, add this to build it for GitHub Pages:')
console.log()
console.log(
' ' +
chalk.green('"homepage"') +
chalk.cyan(': ') +
chalk.green('"http://myname.github.io/myapp"') +
chalk.cyan(','),
)
console.log()
}
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()
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()
}
})
}
function copyPublicFolder() {
fs.copySync(paths.appPublic, paths.appBuild, {
dereference: true,
filter: file => file !== paths.appHtml,
})
}

View File

@@ -1,355 +0,0 @@
process.env.NODE_ENV = 'development'
// Load environment variables from .env file. Suppress warnings using silent
// if this file is missing. dotenv will never modify any environment variables
// that have already been set.
// https://github.com/motdotla/dotenv
require('dotenv').config({ silent: true })
var chalk = require('chalk')
var webpack = require('webpack')
var WebpackDevServer = require('webpack-dev-server')
var historyApiFallback = require('connect-history-api-fallback')
var httpProxyMiddleware = require('http-proxy-middleware')
var detect = require('detect-port')
var clearConsole = require('react-dev-utils/clearConsole')
var checkRequiredFiles = require('react-dev-utils/checkRequiredFiles')
var formatWebpackMessages = require('react-dev-utils/formatWebpackMessages')
var getProcessForPort = require('react-dev-utils/getProcessForPort')
var openBrowser = require('react-dev-utils/openBrowser')
var prompt = require('react-dev-utils/prompt')
var fs = require('fs')
var config = require('../config/webpack.config.dev')
var paths = require('../config/paths')
var useYarn = fs.existsSync(paths.yarnLockFile)
var cli = useYarn ? 'yarn' : 'npm'
var isInteractive = process.stdout.isTTY
// Warn and crash if required files are missing
if (!checkRequiredFiles([paths.appHtml, paths.appIndexJs])) {
process.exit(1)
}
// Tools like Cloud9 rely on this.
var DEFAULT_PORT = process.env.PORT || 3000
var compiler
var handleCompile
// You can safely remove this after ejecting.
// We only use this block for testing of Create React App itself:
var isSmokeTest = process.argv.some(arg => arg.indexOf('--smoke-test') > -1)
if (isSmokeTest) {
handleCompile = function(err, stats) {
if (err || stats.hasErrors() || stats.hasWarnings()) {
process.exit(1)
} else {
process.exit(0)
}
}
}
function setupCompiler(host, port, protocol) {
// "Compiler" is a low-level interface to Webpack.
// It lets us listen to some events and provide our own custom messages.
compiler = webpack(config, handleCompile)
// "invalid" event fires when you have changed a file, and Webpack is
// recompiling a bundle. WebpackDevServer takes care to pause serving the
// bundle, so if you refresh, it'll wait instead of serving the old one.
// "invalid" is short for "bundle invalidated", it doesn't imply any errors.
compiler.plugin('invalid', function() {
if (isInteractive) {
clearConsole()
}
console.log('Compiling...')
})
var isFirstCompile = true
// "done" event fires when Webpack has finished recompiling the bundle.
// Whether or not you have warnings or errors, you will get this event.
compiler.plugin('done', function(stats) {
if (isInteractive) {
clearConsole()
}
// We have switched off the default Webpack output in WebpackDevServer
// options so we are going to "massage" the warnings and errors and present
// them in a readable focused way.
var messages = formatWebpackMessages(stats.toJson({}, true))
var isSuccessful = !messages.errors.length && !messages.warnings.length
var showInstructions = isSuccessful && (isInteractive || isFirstCompile)
if (isSuccessful) {
console.log(chalk.green('Compiled successfully!'))
}
if (showInstructions) {
console.log()
console.log('The app is running at:')
console.log()
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(cli + ' run build') +
'.',
)
console.log()
isFirstCompile = false
}
// If errors exist, only show errors.
if (messages.errors.length) {
console.log(chalk.red('Failed to compile.'))
console.log()
messages.errors.forEach(message => {
console.log(message)
console.log()
})
return
}
// Show warnings if no errors were found.
if (messages.warnings.length) {
console.log(chalk.yellow('Compiled with warnings.'))
console.log()
messages.warnings.forEach(message => {
console.log(message)
console.log()
})
// Teach some ESLint tricks.
console.log('You may use special comments to disable some warnings.')
console.log(
'Use ' +
chalk.yellow('// eslint-disable-next-line') +
' to ignore the next line.',
)
console.log(
'Use ' +
chalk.yellow('/* eslint-disable */') +
' to ignore all warnings in a file.',
)
}
})
}
// We need to provide a custom onError function for httpProxyMiddleware.
// It allows us to log custom error messages on the console.
function onProxyError(proxy) {
return function(err, req, res) {
var host = req.headers && req.headers.host
console.log(
chalk.red('Proxy error:') +
' Could not proxy request ' +
chalk.cyan(req.url) +
' from ' +
chalk.cyan(host) +
' to ' +
chalk.cyan(proxy) +
'.',
)
console.log(
'See https://nodejs.org/api/errors.html#errors_common_system_errors for more information (' +
chalk.cyan(err.code) +
').',
)
console.log()
// And immediately send the proper error response to the client.
// Otherwise, the request will eventually timeout with ERR_EMPTY_RESPONSE on the client side.
if (res.writeHead && !res.headersSent) {
res.writeHead(500)
}
res.end(
'Proxy error: Could not proxy request ' +
req.url +
' from ' +
host +
' to ' +
proxy +
' (' +
err.code +
').',
)
}
}
function addMiddleware(devServer) {
// `proxy` lets you to specify a fallback server during development.
// Every unrecognized request will be forwarded to it.
var proxy = require(paths.appPackageJson).proxy
devServer.use(
historyApiFallback({
// Paths with dots should still use the history fallback.
// See https://github.com/facebookincubator/create-react-app/issues/387.
disableDotRule: true,
// For single page apps, we generally want to fallback to /index.html.
// However we also want to respect `proxy` for API calls.
// So if `proxy` is specified, we need to decide which fallback to use.
// We use a heuristic: if request `accept`s text/html, we pick /index.html.
// Modern browsers include text/html into `accept` header when navigating.
// However API calls like `fetch()` wont generally accept text/html.
// If this heuristic doesnt work well for you, dont use `proxy`.
htmlAcceptHeaders: proxy ? ['text/html'] : ['text/html', '*/*'],
}),
)
if (proxy) {
if (typeof proxy !== 'string') {
console.log(
chalk.red('When specified, "proxy" in package.json must be a string.'),
)
console.log(
chalk.red('Instead, the type of "proxy" was "' + typeof proxy + '".'),
)
console.log(
chalk.red(
'Either remove "proxy" from package.json, or make it a string.',
),
)
process.exit(1)
}
// Otherwise, if proxy is specified, we will let it handle any request.
// There are a few exceptions which we won't send to the proxy:
// - /index.html (served as HTML5 history API fallback)
// - /*.hot-update.json (WebpackDevServer uses this too for hot reloading)
// - /sockjs-node/* (WebpackDevServer uses this for hot reloading)
// Tip: use https://jex.im/regulex/ to visualize the regex
var mayProxy = /^(?!\/(index\.html$|.*\.hot-update\.json$|sockjs-node\/)).*$/
// Pass the scope regex both to Express and to the middleware for proxying
// of both HTTP and WebSockets to work without false positives.
var hpm = httpProxyMiddleware(pathname => mayProxy.test(pathname), {
target: proxy,
logLevel: 'silent',
onProxyReq: function(proxyReq, req, res) {
// Browers may send Origin headers even with same-origin
// requests. To prevent CORS issues, we have to change
// the Origin to match the target URL.
if (proxyReq.getHeader('origin')) {
proxyReq.setHeader('origin', proxy)
}
},
onError: onProxyError(proxy),
secure: false,
changeOrigin: true,
ws: true,
})
devServer.use(mayProxy, hpm)
// Listen for the websocket 'upgrade' event and upgrade the connection.
// If this is not done, httpProxyMiddleware will not try to upgrade until
// an initial plain HTTP request is made.
devServer.listeningApp.on('upgrade', hpm.upgrade)
}
// Finally, by now we have certainly resolved the URL.
// It may be /index.html, so let the dev server try serving it again.
devServer.use(devServer.middleware)
}
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',
// By default WebpackDevServer serves physical files from current directory
// in addition to all the virtual build products that it serves from memory.
// This is confusing because those files wont automatically be available in
// production build folder unless we copy them. However, copying the whole
// project directory is dangerous because we may expose sensitive files.
// Instead, we establish a convention that only files in `public` directory
// get served. Our build script will copy `public` into the `build` folder.
// In `index.html`, you can get URL of `public` folder with %PUBLIC_PATH%:
// <link rel="shortcut icon" href="%PUBLIC_URL%/favicon.ico">
// In JavaScript code, you can access it with `process.env.PUBLIC_URL`.
// Note that we only recommend to use `public` folder as an escape hatch
// for files like `favicon.ico`, `manifest.json`, and libraries that are
// for some reason broken when imported through Webpack. If you just want to
// use an image, put it in `src` and `import` it from JavaScript instead.
contentBase: paths.appPublic,
// Enable hot reloading server. It will provide /sockjs-node/ endpoint
// for the WebpackDevServer client so it can learn when the files were
// updated. The WebpackDevServer client is included as an entry point
// in the Webpack development configuration. Note that only changes
// to CSS are currently hot reloaded. JS changes will refresh the browser.
hot: true,
// It is important to tell WebpackDevServer to use the same "root" path
// as we specified in the config. In development, we always serve from /.
publicPath: config.output.publicPath,
// WebpackDevServer is noisy by default so we emit custom message instead
// by listening to the compiler events with `compiler.plugin` calls above.
quiet: true,
// Reportedly, this avoids CPU overload on some systems.
// https://github.com/facebookincubator/create-react-app/issues/293
watchOptions: {
ignored: /node_modules/,
},
// Enable HTTPS if the HTTPS environment variable is set to 'true'
https: protocol === 'https',
host: host,
})
// Our custom middleware proxies requests to /index.html or a remote API.
addMiddleware(devServer)
// Launch WebpackDevServer.
devServer.listen(port, (err, result) => {
if (err) {
return console.log(err)
}
if (isInteractive) {
clearConsole()
}
console.log(chalk.cyan('Starting the development server...'))
console.log()
if (isInteractive && process.env.OPEN_BROWSER !== 'false') {
openBrowser(protocol + '://' + host + ':' + port + '/')
}
})
}
function run(port) {
var protocol = process.env.HTTPS === 'true' ? 'https' : 'http'
var host = process.env.HOST || 'localhost'
setupCompiler(host, port, protocol)
runDevServer(host, port, protocol)
}
// We attempt to use the default port but if it is busy, we offer the user to
// run on a different port. `detect()` Promise resolves to the next free port.
detect(DEFAULT_PORT).then(port => {
if (port === DEFAULT_PORT) {
run(port)
return
}
if (isInteractive) {
clearConsole()
var existingProcess = getProcessForPort(DEFAULT_PORT)
var question =
chalk.yellow(
'Something is already running on port ' +
DEFAULT_PORT +
'.' +
(existingProcess ? ' Probably:\n ' + existingProcess : ''),
) + '\n\nWould you like to run the app on another port instead?'
prompt(question, true).then(shouldChangePort => {
if (shouldChangePort) {
run(port)
}
})
} else {
console.log(
chalk.red('Something is already running on port ' + DEFAULT_PORT + '.'),
)
}
})

View File

@@ -1,18 +0,0 @@
process.env.NODE_ENV = 'test'
process.env.PUBLIC_URL = ''
// Load environment variables from .env file. Suppress warnings using silent
// if this file is missing. dotenv will never modify any environment variables
// that have already been set.
// https://github.com/motdotla/dotenv
require('dotenv').config({ silent: true })
const jest = require('jest')
const argv = process.argv.slice(2)
// Watch unless on CI or in coverage mode
if (!process.env.CI && argv.indexOf('--coverage') < 0) {
argv.push('--watch')
}
jest.run(argv)

1
web/src/index.js Normal file
View File

@@ -0,0 +1 @@
import '../../index.web'

View File

@@ -0,0 +1,51 @@
// In production, we register a service worker to serve assets from local cache.
// This lets the app load faster on subsequent visits in production, and gives
// it offline capabilities. However, it also means that developers (and users)
// will only see deployed updates on the "N+1" visit to a page, since previously
// cached resources are updated in the background.
// To learn more about the benefits of this model, read https://goo.gl/KwvDNy.
// This link also includes instructions on opting out of this behavior.
export default function register() {
if (process.env.NODE_ENV === 'production' && 'serviceWorker' in navigator) {
window.addEventListener('load', () => {
const swUrl = `${process.env.PUBLIC_URL}/service-worker.js`
navigator.serviceWorker
.register(swUrl)
.then(registration => {
registration.onupdatefound = () => {
const installingWorker = registration.installing
installingWorker.onstatechange = () => {
if (installingWorker.state === 'installed') {
if (navigator.serviceWorker.controller) {
// At this point, the old content will have been purged and
// the fresh content will have been added to the cache.
// It's the perfect time to display a "New content is
// available; please refresh." message in your web app.
console.log('New content is available; please refresh.')
} else {
// At this point, everything has been precached.
// It's the perfect time to display a
// "Content is cached for offline use." message.
console.log('Content is cached for offline use.')
}
}
}
}
})
.catch(error => {
console.error('Error during service worker registration:', error)
})
})
}
}
export function unregister() {
if ('serviceWorker' in navigator) {
navigator.serviceWorker.ready.then(registration => {
registration.unregister()
})
}
}

File diff suppressed because it is too large Load Diff