react/scripts/error-codes/transform-error-messages.js
Andrew Clark 875d05d553
Include full error messages in React Native build (#15363)
The React Native build does not minify error messages in production,
but it still needs to run the error messages transform to compile
`invariant` calls to `ReactError`. To do this, I added a `noMinify`
option to the Babel plugin. I also renamed it from
`minify-error-messages` to the more generic `transform-error-messages`.
2019-04-09 16:40:19 -07:00

130 lines
4.2 KiB
JavaScript

/**
* Copyright (c) Facebook, Inc. and its affiliates.
*
* This source code is licensed under the MIT license found in the
* LICENSE file in the root directory of this source tree.
*/
'use strict';
const fs = require('fs');
const evalToString = require('../shared/evalToString');
const invertObject = require('./invertObject');
module.exports = function(babel) {
const t = babel.types;
const DEV_EXPRESSION = t.identifier('__DEV__');
return {
visitor: {
CallExpression(path, file) {
const node = path.node;
const noMinify = file.opts.noMinify;
if (path.get('callee').isIdentifier({name: 'invariant'})) {
// Turns this code:
//
// invariant(condition, 'A %s message that contains %s', adj, noun);
//
// into this:
//
// if (!condition) {
// if (__DEV__) {
// throw ReactError(`A ${adj} message that contains ${noun}`);
// } else {
// throw ReactErrorProd(ERR_CODE, adj, noun);
// }
// }
//
// where ERR_CODE is an error code: a unique identifier (a number
// string) that references a verbose error message. The mapping is
// stored in `scripts/error-codes/codes.json`.
const condition = node.arguments[0];
const errorMsgLiteral = evalToString(node.arguments[1]);
const errorMsgExpressions = Array.from(node.arguments.slice(2));
const errorMsgQuasis = errorMsgLiteral
.split('%s')
.map(raw => t.templateElement({raw, cooked: String.raw({raw})}));
// Import ReactError
const reactErrorIdentfier = file.addImport(
'shared/ReactError',
'default',
'ReactError'
);
// Outputs:
// throw ReactError(`A ${adj} message that contains ${noun}`);
const devThrow = t.throwStatement(
t.callExpression(reactErrorIdentfier, [
t.templateLiteral(errorMsgQuasis, errorMsgExpressions),
])
);
// Avoid caching because we write it as we go.
const existingErrorMap = JSON.parse(
fs.readFileSync(__dirname + '/codes.json', 'utf-8')
);
const errorMap = invertObject(existingErrorMap);
let prodErrorId = errorMap[errorMsgLiteral];
if (prodErrorId === undefined || noMinify) {
// There is no error code for this message. We use a lint rule to
// enforce that messages can be minified, so assume this is
// intentional and exit gracefully.
//
// Outputs:
// if (!condition) {
// throw ReactError(`A ${adj} message that contains ${noun}`);
// }
path.replaceWith(
t.ifStatement(
t.unaryExpression('!', condition),
t.blockStatement([devThrow])
)
);
return;
}
prodErrorId = parseInt(prodErrorId, 10);
// Import ReactErrorProd
const reactErrorProdIdentfier = file.addImport(
'shared/ReactErrorProd',
'default',
'ReactErrorProd'
);
// Outputs:
// throw ReactErrorProd(ERR_CODE, adj, noun);
const prodThrow = t.throwStatement(
t.callExpression(reactErrorProdIdentfier, [
t.numericLiteral(prodErrorId),
...errorMsgExpressions,
])
);
// Outputs:
// if (!condition) {
// if (__DEV__) {
// throw ReactError(`A ${adj} message that contains ${noun}`);
// } else {
// throw ReactErrorProd(ERR_CODE, adj, noun);
// }
// }
path.replaceWith(
t.ifStatement(
t.unaryExpression('!', condition),
t.blockStatement([
t.ifStatement(
DEV_EXPRESSION,
t.blockStatement([devThrow]),
t.blockStatement([prodThrow])
),
])
)
);
}
},
},
};
};