UNPKG

babel-plugin-debug-macros

Version:
213 lines 23.5 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); class Builder { constructor(t, util, options) { this.t = t; this.expressions = []; 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); } this.expressions.push([ path, this._buildLogicalExpressions(prefixedIdentifiers, callExpression), ]); } /** * 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.`); } }, }); } /** * Performs the actually expansion of macros */ expandMacros() { let t = this.t; for (let i = 0; i < this.expressions.length; i++) { let expression = this.expressions[i]; let exp = expression[0]; let flag = this._debugExpression(exp); let logicalExp = expression[1]; exp.replaceWith(t.parenthesizedExpression(logicalExp(flag))); exp.scope.crawl(); } } _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) { let t = this.t; return (debugIdentifier) => { 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 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  private expressions: [CallStatementPath, (debugIdentifier: t.Expression) => t.Expression][] = [];\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    this.expressions.push([\n      path,\n      this._buildLogicalExpressions(prefixedIdentifiers, callExpression),\n    ]);\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  /**\n   * Performs the actually expansion of macros\n   */\n  expandMacros() {\n    let t = this.t;\n    for (let i = 0; i < this.expressions.length; i++) {\n      let expression = this.expressions[i];\n      let exp = expression[0];\n      let flag = this._debugExpression(exp);\n      let logicalExp = expression[1];\n      exp.replaceWith(t.parenthesizedExpression(logicalExp(flag)));\n      exp.scope.crawl();\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) => t.Expression {\n    let t = this.t;\n\n    return (debugIdentifier: t.Expression) => {\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 logicalExpressions!;\n    };\n  }\n}\n"]}