babel-plugin-debug-macros
Version:
Debug macros and feature flag stripping
196 lines • 21.7 kB
JavaScript
;
Object.defineProperty(exports, "__esModule", { value: true });
class Builder {
constructor(t, util, options) {
this.t = t;
this.module = options.module;
this.global = options.global;
this.assertPredicateIndex = options.assertPredicateIndex;
this.isDebug = options.isDebug;
this.util = util;
}
/**
* Expands:
*
* assert($PREDICATE, $MESSAGE)
*
* into
*
* ($DEBUG && console.assert($PREDICATE, $MESSAGE));
*
* or
*
* ($DEBUG && assert($PREDICATE, $MESSAGE));
*
* or
*
* ($DEBUG && $GLOBAL_NS.assert($PREDICATE, $MESSAGE));
*/
assert(path) {
let predicate;
const index = this.assertPredicateIndex;
if (index !== undefined) {
predicate = (expression, args) => {
return args[index];
};
}
this._createMacroExpression(path, {
predicate,
});
}
/**
* Expands:
*
* warn($MESSAGE)
*
* into
*
* ($DEBUG && console.warn($MESSAGE));
*
* or
*
* ($DEBUG && warn($MESSAGE));
*
* or
*
* ($DEBUG && $GLOBAL_NS.warn($MESSAGE));
*/
warn(path) {
this._createMacroExpression(path);
}
/**
* Expands:
*
* log($MESSAGE)
*
* into
*
* ($DEBUG && console.log($MESSAGE));
*
* or
*
* ($DEBUG && log($MESSAGE));
*
* or
*
* ($DEBUG && $GLOBAL_NS.log($MESSAGE));
*/
log(path) {
this._createMacroExpression(path);
}
_createMacroExpression(path, options = {}) {
let t = this.t;
let expression = path.node.expression;
let callee = expression.callee;
let args = expression.arguments;
if (options.validate) {
options.validate(expression, args);
}
let callExpression;
if (this.module) {
callExpression = expression;
}
else if (this.global) {
callExpression = this._createGlobalExternalHelper(callee, args, this.global);
}
else if (options.buildConsoleAPI) {
callExpression = options.buildConsoleAPI(expression, args);
}
else {
callExpression = this._createConsoleAPI(options.consoleAPI || callee, args);
}
let prefixedIdentifiers = [];
if (options.predicate) {
let predicate = options.predicate(expression, args) || t.identifier('false');
if (!this.t.isExpression(predicate)) {
throw new Error(`bug: this doesn't support ${predicate.type}`);
}
let negatedPredicate = t.unaryExpression('!', t.parenthesizedExpression(predicate));
prefixedIdentifiers.push(negatedPredicate);
}
// Expand the macro!
let replacementPath = this._buildLogicalExpressions(prefixedIdentifiers, callExpression, this._debugExpression(path));
path.replaceWith(replacementPath);
path.scope.crawl();
}
/**
* Expands:
*
* deprecate($MESSAGE, $PREDICATE)
*
* or
*
* deprecate($MESSAGE, $PREDICATE, {
* $ID,
* $URL,
* $UNIL
* });
*
* into
*
* ($DEBUG && $PREDICATE && console.warn($MESSAGE));
*
* or
*
* ($DEBUG && $PREDICATE && deprecate($MESSAGE, $PREDICATE, { $ID, $URL, $UNTIL }));
*
* or
*
* ($DEBUG && $PREDICATE && $GLOBAL_NS.deprecate($MESSAGE, $PREDICATE, { $ID, $URL, $UNTIL }));
*/
deprecate(path) {
this._createMacroExpression(path, {
predicate: (expression, args) => args[1],
buildConsoleAPI: (expression, args) => {
let message = args[0];
return this._createConsoleAPI(this.t.identifier('warn'), [message]);
},
validate: (expression, args) => {
let meta = args[2];
if (meta &&
this.t.isObjectExpression(meta) &&
meta.properties &&
!meta.properties.some((prop) => this.t.isObjectProperty(prop) &&
((this.t.isIdentifier(prop.key) && prop.key.name === 'id') ||
(this.t.isStringLiteral(prop.key) && prop.key.value === 'id')))) {
throw new ReferenceError(`deprecate's meta information requires an "id" field.`);
}
},
});
}
_debugExpression(target) {
if (typeof this.isDebug === 'boolean') {
return this.t.booleanLiteral(this.isDebug);
}
else {
return this.t.callExpression(this.util.import(target, '@embroider/macros', 'isDevelopingApp'), []);
}
}
_createGlobalExternalHelper(identifier, args, ns) {
let t = this.t;
return t.callExpression(t.memberExpression(t.identifier(ns), identifier), args);
}
_createConsoleAPI(identifier, args) {
let t = this.t;
return t.callExpression(t.memberExpression(t.identifier('console'), identifier), args);
}
_buildLogicalExpressions(identifiers, callExpression, debugIdentifier) {
let t = this.t;
identifiers.unshift(debugIdentifier);
identifiers.push(callExpression);
let logicalExpressions;
for (let i = 0; i < identifiers.length; i++) {
let left = identifiers[i];
let right = identifiers[i + 1];
if (!logicalExpressions) {
logicalExpressions = t.logicalExpression('&&', left, right);
}
else if (right) {
logicalExpressions = t.logicalExpression('&&', logicalExpressions, right);
}
}
return t.parenthesizedExpression(logicalExpressions);
}
}
exports.default = Builder;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"builder.js","sourceRoot":"","sources":["../../../src/utils/builder.ts"],"names":[],"mappings":"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","sourcesContent":["import type * as Babel from '@babel/core';\nimport type { NodePath, types as t } from '@babel/core';\nimport { CallIdentifierExpression, CallStatementPath } from './babel-type-helpers';\nimport { ImportUtil } from 'babel-import-util';\n\nexport interface Options {\n  module: boolean | undefined;\n  global: string | undefined;\n  assertPredicateIndex: number | undefined;\n  isDebug: boolean | \"@embroider/macros\";\n}\n\ninterface MacroExpressionOpts {\n  validate?: (expression: CallIdentifierExpression, args: t.CallExpression['arguments']) => void;\n  buildConsoleAPI?: (\n    expression: CallIdentifierExpression,\n    args: t.CallExpression['arguments']\n  ) => t.CallExpression;\n  consoleAPI?: t.Identifier;\n  predicate?: (\n    expression: CallIdentifierExpression,\n    args: t.CallExpression['arguments']\n  ) => t.CallExpression['arguments'][number] | undefined;\n}\n\nexport default class Builder {\n  private module: boolean | undefined;\n  private global: string | undefined;\n  private assertPredicateIndex: number | undefined;\n  private isDebug: boolean | '@embroider/macros';\n  private util: ImportUtil;\n\n  constructor(\n    readonly t: typeof Babel.types,\n    util: ImportUtil,\n    options: Options\n  ) {\n    this.module = options.module;\n    this.global = options.global;\n    this.assertPredicateIndex = options.assertPredicateIndex;\n    this.isDebug = options.isDebug;\n    this.util = util;\n  }\n\n  /**\n   * Expands:\n   *\n   * assert($PREDICATE, $MESSAGE)\n   *\n   * into\n   *\n   * ($DEBUG && console.assert($PREDICATE, $MESSAGE));\n   *\n   * or\n   *\n   * ($DEBUG && assert($PREDICATE, $MESSAGE));\n   *\n   * or\n   *\n   * ($DEBUG && $GLOBAL_NS.assert($PREDICATE, $MESSAGE));\n   */\n  assert(path: CallStatementPath) {\n    let predicate: MacroExpressionOpts['predicate'];\n    const index = this.assertPredicateIndex;\n    if (index !== undefined) {\n      predicate = (expression, args) => {\n        return args[index];\n      };\n    }\n\n    this._createMacroExpression(path, {\n      predicate,\n    });\n  }\n\n  /**\n   * Expands:\n   *\n   * warn($MESSAGE)\n   *\n   * into\n   *\n   * ($DEBUG && console.warn($MESSAGE));\n   *\n   * or\n   *\n   * ($DEBUG && warn($MESSAGE));\n   *\n   * or\n   *\n   * ($DEBUG && $GLOBAL_NS.warn($MESSAGE));\n   */\n  warn(path: CallStatementPath) {\n    this._createMacroExpression(path);\n  }\n\n  /**\n   * Expands:\n   *\n   * log($MESSAGE)\n   *\n   * into\n   *\n   * ($DEBUG && console.log($MESSAGE));\n   *\n   * or\n   *\n   * ($DEBUG && log($MESSAGE));\n   *\n   * or\n   *\n   * ($DEBUG && $GLOBAL_NS.log($MESSAGE));\n   */\n  log(path: CallStatementPath) {\n    this._createMacroExpression(path);\n  }\n\n  _createMacroExpression(path: CallStatementPath, options: MacroExpressionOpts = {}) {\n    let t = this.t;\n    let expression = path.node.expression;\n    let callee = expression.callee;\n    let args = expression.arguments;\n\n    if (options.validate) {\n      options.validate(expression, args);\n    }\n\n    let callExpression;\n    if (this.module) {\n      callExpression = expression;\n    } else if (this.global) {\n      callExpression = this._createGlobalExternalHelper(callee, args, this.global);\n    } else if (options.buildConsoleAPI) {\n      callExpression = options.buildConsoleAPI(expression, args);\n    } else {\n      callExpression = this._createConsoleAPI(options.consoleAPI || callee, args);\n    }\n\n    let prefixedIdentifiers: t.Expression[] = [];\n\n    if (options.predicate) {\n      let predicate = options.predicate(expression, args) || t.identifier('false');\n      if (!this.t.isExpression(predicate)) {\n        throw new Error(`bug: this doesn't support ${predicate.type}`);\n      }\n      let negatedPredicate = t.unaryExpression('!', t.parenthesizedExpression(predicate));\n      prefixedIdentifiers.push(negatedPredicate);\n    }\n\n    // Expand the macro!\n    let replacementPath = this._buildLogicalExpressions(prefixedIdentifiers, callExpression, this._debugExpression(path));\n    path.replaceWith(replacementPath);\n    path.scope.crawl();\n  }\n\n  /**\n   * Expands:\n   *\n   * deprecate($MESSAGE, $PREDICATE)\n   *\n   * or\n   *\n   * deprecate($MESSAGE, $PREDICATE, {\n   *  $ID,\n   *  $URL,\n   *  $UNIL\n   * });\n   *\n   * into\n   *\n   * ($DEBUG && $PREDICATE && console.warn($MESSAGE));\n   *\n   * or\n   *\n   * ($DEBUG && $PREDICATE && deprecate($MESSAGE, $PREDICATE, { $ID, $URL, $UNTIL }));\n   *\n   * or\n   *\n   * ($DEBUG && $PREDICATE && $GLOBAL_NS.deprecate($MESSAGE, $PREDICATE, { $ID, $URL, $UNTIL }));\n   */\n  deprecate(path: CallStatementPath) {\n    this._createMacroExpression(path, {\n      predicate: (expression, args) => args[1],\n\n      buildConsoleAPI: (expression, args) => {\n        let message = args[0];\n\n        return this._createConsoleAPI(this.t.identifier('warn'), [message]);\n      },\n\n      validate: (expression, args) => {\n        let meta = args[2];\n\n        if (\n          meta &&\n          this.t.isObjectExpression(meta) &&\n          meta.properties &&\n          !meta.properties.some(\n            (prop) =>\n              this.t.isObjectProperty(prop) &&\n              ((this.t.isIdentifier(prop.key) && prop.key.name === 'id') ||\n                (this.t.isStringLiteral(prop.key) && prop.key.value === 'id'))\n          )\n        ) {\n          throw new ReferenceError(`deprecate's meta information requires an \"id\" field.`);\n        }\n      },\n    });\n  }\n\n  _debugExpression(target: NodePath) {\n    if (typeof this.isDebug === 'boolean') {\n      return this.t.booleanLiteral(this.isDebug);\n    } else {\n      return this.t.callExpression(\n        this.util.import(target, '@embroider/macros', 'isDevelopingApp'),\n        []\n      );\n    }\n  }\n\n  _createGlobalExternalHelper(\n    identifier: t.Identifier,\n    args: t.CallExpression['arguments'],\n    ns: string\n  ) {\n    let t = this.t;\n    return t.callExpression(t.memberExpression(t.identifier(ns), identifier), args);\n  }\n\n  _createConsoleAPI(identifier: t.Identifier, args: t.CallExpression['arguments']) {\n    let t = this.t;\n    return t.callExpression(t.memberExpression(t.identifier('console'), identifier), args);\n  }\n\n  _buildLogicalExpressions(\n    identifiers: t.Expression[],\n    callExpression: t.Expression,\n    debugIdentifier: t.Expression\n  ): t.Expression {\n    let t = this.t;\n\n    identifiers.unshift(debugIdentifier);\n    identifiers.push(callExpression);\n    let logicalExpressions;\n\n    for (let i = 0; i < identifiers.length; i++) {\n      let left = identifiers[i];\n      let right = identifiers[i + 1];\n      if (!logicalExpressions) {\n        logicalExpressions = t.logicalExpression('&&', left, right);\n      } else if (right) {\n        logicalExpressions = t.logicalExpression('&&', logicalExpressions, right);\n      }\n    }\n\n    return t.parenthesizedExpression(logicalExpressions!);\n  }\n}\n"]}