Socket
Socket
Sign inDemoInstall

babel-import-util

Package Overview
Dependencies
Maintainers
1
Versions
17
Alerts
File Explorer

Advanced tools

Socket logo

Install Socket

Detect and block malicious and high-risk dependencies

Install

babel-import-util - npm Package Compare versions

Comparing version 0.2.0 to 1.0.0

src/CHANGELOG.md

2

package.json
{
"name": "babel-import-util",
"version": "0.2.0",
"version": "1.0.0",
"description": "Utility for manipulating imports within babel plugins",

@@ -5,0 +5,0 @@ "repository": "https://github.com/ef4/babel-import-util",

# babel-import-util
Makes it easier for a babel plugin to emit imported names. Key benefits:
- the output composes correctly with subsequent babel plugins, because we update Babel's understanding of the bindings
- redundant imports will be deduplicated automatically
- written in TypeScript
## Usage by example:
If you want to rewrite:
```js
myTarget('hello world');
```
To:
```js
import { theMethod } from 'my-implementation';
theMethod('hello world');
```
Your plugin would look like this:
```js
function testTransform(babel) {
return {
visitor: {
Program: {
enter(path, state) {
// Always instantiate the ImportUtil instance at the Program scope
state.importUtil = new ImportUtil(babel.types, path);
},
},
CallExpression(path, state) {
let callee = path.get('callee');
if (callee.isIdentifier() && callee.node.name === 'myTarget') {
callee.replaceWith(state.importUtil.import(callee, 'my-implementation', 'theMethod'));
}
},
},
};
}
```
## API
```ts
import type { NodePath } from '@babel/traverse';
import type * as t from '@babel/types';
class ImportUtil {
// Import the given value (if needed) and return an Identifier representing
// it.
import(
// the spot at which you will insert the Identifier we return to you
target: NodePath<t.Node>,
// the path to the module you're importing from
moduleSpecifier: string,
// the name you're importing from that module. Use "default" for the default
// export. Use "*" for the namespace.
exportedName: string,
// Optional hint for helping us pick a name for the imported binding
nameHint?: string
): t.Identifier;
// Remove an import specifier. If the removed specifier is
// the last one on the whole import statement, the whole
// statement is also removed.
//
// You can use "default" and "*" as exportedName to handle
// those special cases.
removeImport(moduleSpecifier: string, exportedName: string): void;
}
```

@@ -11,3 +11,4 @@ import type { NodePath } from '@babel/traverse';

private addSpecifier;
private buildSpecifier;
}
export {};

@@ -17,6 +17,3 @@ "use strict";

.get('specifiers')
.find((specifierPath) => exportedName === 'default'
? specifierPath.isImportDefaultSpecifier()
: specifierPath.isImportSpecifier() &&
name(specifierPath.node.imported) === exportedName);
.find((specifierPath) => matchSpecifier(specifierPath, exportedName));
if (importSpecifierPath) {

@@ -39,4 +36,4 @@ if (topLevelPath.node.specifiers.length === 1) {

moduleSpecifier,
// the name you're importing from that module (use "default" for the default
// export)
// the name you're importing from that module. Use "default" for the default
// export. Use "*" for the namespace.
exportedName,

@@ -52,5 +49,3 @@ // Optional hint for helping us pick a name for the imported binding

.get('specifiers')
.find((spec) => exportedName === 'default'
? spec.isImportDefaultSpecifier()
: spec.isImportSpecifier() && name(spec.node.imported) === exportedName);
.find((spec) => matchSpecifier(spec, exportedName));
if (specifier && ((_a = target.scope.getBinding(specifier.node.local.name)) === null || _a === void 0 ? void 0 : _a.kind) === 'module') {

@@ -70,5 +65,3 @@ return this.t.identifier(specifier.node.local.name);

let local = this.t.identifier(unusedNameLike(target, desiredName(nameHint, exportedName, target)));
let specifier = exportedName === 'default'
? this.t.importDefaultSpecifier(local)
: this.t.importSpecifier(local, this.t.identifier(exportedName));
let specifier = this.buildSpecifier(exportedName, local);
declaration.node.specifiers.push(specifier);

@@ -78,2 +71,12 @@ declaration.scope.registerBinding('module', declaration.get(`specifiers.${declaration.node.specifiers.length - 1}`));

}
buildSpecifier(exportedName, localName) {
switch (exportedName) {
case 'default':
return this.t.importDefaultSpecifier(localName);
case '*':
return this.t.importNamespaceSpecifier(localName);
default:
return this.t.importSpecifier(localName, this.t.identifier(exportedName));
}
}
}

@@ -101,3 +104,3 @@ exports.ImportUtil = ImportUtil;

}
if (exportedName === 'default') {
if (exportedName === 'default' || exportedName === '*') {
if (target.isIdentifier()) {

@@ -114,2 +117,12 @@ return target.node.name;

}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["index.ts"],"names":[],"mappings":"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","sourcesContent":["import type { NodePath } from '@babel/traverse';\nimport type * as t from '@babel/types';\n\ntype BabelTypes = typeof t;\n\nexport class ImportUtil {\n  constructor(private t: BabelTypes, private program: NodePath<t.Program>) {}\n\n  removeImport(moduleSpecifier: string, exportedName: string) {\n    for (let topLevelPath of this.program.get('body')) {\n      if (\n        !topLevelPath.isImportDeclaration() ||\n        topLevelPath.get('source').node.value !== moduleSpecifier\n      ) {\n        continue;\n      }\n\n      let importSpecifierPath = topLevelPath\n        .get('specifiers')\n        .find((specifierPath) =>\n          exportedName === 'default'\n            ? specifierPath.isImportDefaultSpecifier()\n            : specifierPath.isImportSpecifier() &&\n              name(specifierPath.node.imported) === exportedName\n        );\n      if (importSpecifierPath) {\n        if (topLevelPath.node.specifiers.length === 1) {\n          topLevelPath.remove();\n        } else {\n          importSpecifierPath.remove();\n        }\n      }\n    }\n  }\n\n  // Import the given value (if needed) and return an Identifier representing\n  // it.\n  import(\n    // the spot at which you will insert the Identifier we return to you\n    target: NodePath<t.Node>,\n\n    // the path to the module you're importing from\n    moduleSpecifier: string,\n\n    // the name you're importing from that module (use \"default\" for the default\n    // export)\n    exportedName: string,\n\n    // Optional hint for helping us pick a name for the imported binding\n    nameHint?: string\n  ): t.Identifier {\n    let declaration = this.program\n      .get('body')\n      .find((elt) => elt.isImportDeclaration() && elt.node.source.value === moduleSpecifier) as\n      | undefined\n      | NodePath<t.ImportDeclaration>;\n    if (declaration) {\n      let specifier = declaration\n        .get('specifiers')\n        .find((spec) =>\n          exportedName === 'default'\n            ? spec.isImportDefaultSpecifier()\n            : spec.isImportSpecifier() && name(spec.node.imported) === exportedName\n        ) as undefined | NodePath<t.ImportSpecifier> | NodePath<t.ImportDefaultSpecifier>;\n      if (specifier && target.scope.getBinding(specifier.node.local.name)?.kind === 'module') {\n        return this.t.identifier(specifier.node.local.name);\n      } else {\n        return this.addSpecifier(target, declaration, exportedName, nameHint);\n      }\n    } else {\n      this.program.node.body.unshift(\n        this.t.importDeclaration([], this.t.stringLiteral(moduleSpecifier))\n      );\n      return this.addSpecifier(\n        target,\n        this.program.get(`body.0`) as NodePath<t.ImportDeclaration>,\n        exportedName,\n        nameHint\n      );\n    }\n  }\n\n  private addSpecifier(\n    target: NodePath<t.Node>,\n    declaration: NodePath<t.ImportDeclaration>,\n    exportedName: string,\n    nameHint: string | undefined\n  ): t.Identifier {\n    let local = this.t.identifier(\n      unusedNameLike(target, desiredName(nameHint, exportedName, target))\n    );\n    let specifier =\n      exportedName === 'default'\n        ? this.t.importDefaultSpecifier(local)\n        : this.t.importSpecifier(local, this.t.identifier(exportedName));\n    declaration.node.specifiers.push(specifier);\n    declaration.scope.registerBinding(\n      'module',\n      declaration.get(`specifiers.${declaration.node.specifiers.length - 1}`) as NodePath\n    );\n    return local;\n  }\n}\n\nfunction unusedNameLike(path: NodePath<t.Node>, name: string): string {\n  let candidate = name;\n  let counter = 0;\n  while (path.scope.hasBinding(candidate)) {\n    candidate = `${name}${counter++}`;\n  }\n  return candidate;\n}\n\nfunction name(node: t.StringLiteral | t.Identifier): string {\n  if (node.type === 'StringLiteral') {\n    return node.value;\n  } else {\n    return node.name;\n  }\n}\n\nfunction desiredName(nameHint: string | undefined, exportedName: string, target: NodePath<t.Node>) {\n  if (nameHint) {\n    return nameHint;\n  }\n  if (exportedName === 'default') {\n    if (target.isIdentifier()) {\n      return target.node.name;\n    } else {\n      return target.scope.generateUidIdentifierBasedOnNode(target.node).name;\n    }\n  } else {\n    return exportedName;\n  }\n}\n"]}
function matchSpecifier(spec, exportedName) {
switch (exportedName) {
case 'default':
return spec.isImportDefaultSpecifier();
case '*':
return spec.isImportNamespaceSpecifier();
default:
return spec.isImportSpecifier() && name(spec.node.imported) === exportedName;
}
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["index.ts"],"names":[],"mappings":"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","sourcesContent":["import type { NodePath } from '@babel/traverse';\nimport type * as t from '@babel/types';\n\ntype BabelTypes = typeof t;\n\nexport class ImportUtil {\n  constructor(private t: BabelTypes, private program: NodePath<t.Program>) {}\n\n  removeImport(moduleSpecifier: string, exportedName: string) {\n    for (let topLevelPath of this.program.get('body')) {\n      if (\n        !topLevelPath.isImportDeclaration() ||\n        topLevelPath.get('source').node.value !== moduleSpecifier\n      ) {\n        continue;\n      }\n\n      let importSpecifierPath = topLevelPath\n        .get('specifiers')\n        .find((specifierPath) => matchSpecifier(specifierPath, exportedName));\n      if (importSpecifierPath) {\n        if (topLevelPath.node.specifiers.length === 1) {\n          topLevelPath.remove();\n        } else {\n          importSpecifierPath.remove();\n        }\n      }\n    }\n  }\n\n  // Import the given value (if needed) and return an Identifier representing\n  // it.\n  import(\n    // the spot at which you will insert the Identifier we return to you\n    target: NodePath<t.Node>,\n\n    // the path to the module you're importing from\n    moduleSpecifier: string,\n\n    // the name you're importing from that module. Use \"default\" for the default\n    // export. Use \"*\" for the namespace.\n    exportedName: string,\n\n    // Optional hint for helping us pick a name for the imported binding\n    nameHint?: string\n  ): t.Identifier {\n    let declaration = this.program\n      .get('body')\n      .find((elt) => elt.isImportDeclaration() && elt.node.source.value === moduleSpecifier) as\n      | undefined\n      | NodePath<t.ImportDeclaration>;\n    if (declaration) {\n      let specifier = declaration\n        .get('specifiers')\n        .find((spec) => matchSpecifier(spec, exportedName));\n      if (specifier && target.scope.getBinding(specifier.node.local.name)?.kind === 'module') {\n        return this.t.identifier(specifier.node.local.name);\n      } else {\n        return this.addSpecifier(target, declaration, exportedName, nameHint);\n      }\n    } else {\n      this.program.node.body.unshift(\n        this.t.importDeclaration([], this.t.stringLiteral(moduleSpecifier))\n      );\n      return this.addSpecifier(\n        target,\n        this.program.get(`body.0`) as NodePath<t.ImportDeclaration>,\n        exportedName,\n        nameHint\n      );\n    }\n  }\n\n  private addSpecifier(\n    target: NodePath<t.Node>,\n    declaration: NodePath<t.ImportDeclaration>,\n    exportedName: string,\n    nameHint: string | undefined\n  ): t.Identifier {\n    let local = this.t.identifier(\n      unusedNameLike(target, desiredName(nameHint, exportedName, target))\n    );\n    let specifier = this.buildSpecifier(exportedName, local);\n    declaration.node.specifiers.push(specifier);\n    declaration.scope.registerBinding(\n      'module',\n      declaration.get(`specifiers.${declaration.node.specifiers.length - 1}`) as NodePath\n    );\n    return local;\n  }\n\n  private buildSpecifier(exportedName: string, localName: t.Identifier) {\n    switch (exportedName) {\n      case 'default':\n        return this.t.importDefaultSpecifier(localName);\n      case '*':\n        return this.t.importNamespaceSpecifier(localName);\n      default:\n        return this.t.importSpecifier(localName, this.t.identifier(exportedName));\n    }\n  }\n}\n\nfunction unusedNameLike(path: NodePath<t.Node>, name: string): string {\n  let candidate = name;\n  let counter = 0;\n  while (path.scope.hasBinding(candidate)) {\n    candidate = `${name}${counter++}`;\n  }\n  return candidate;\n}\n\nfunction name(node: t.StringLiteral | t.Identifier): string {\n  if (node.type === 'StringLiteral') {\n    return node.value;\n  } else {\n    return node.name;\n  }\n}\n\nfunction desiredName(nameHint: string | undefined, exportedName: string, target: NodePath<t.Node>) {\n  if (nameHint) {\n    return nameHint;\n  }\n  if (exportedName === 'default' || exportedName === '*') {\n    if (target.isIdentifier()) {\n      return target.node.name;\n    } else {\n      return target.scope.generateUidIdentifierBasedOnNode(target.node).name;\n    }\n  } else {\n    return exportedName;\n  }\n}\n\nfunction matchSpecifier(spec: NodePath<any>, exportedName: string): boolean {\n  switch (exportedName) {\n    case 'default':\n      return spec.isImportDefaultSpecifier();\n    case '*':\n      return spec.isImportNamespaceSpecifier();\n    default:\n      return spec.isImportSpecifier() && name(spec.node.imported) === exportedName;\n  }\n}\n"]}
SocketSocket SOC 2 Logo

Product

  • Package Alerts
  • Integrations
  • Docs
  • Pricing
  • FAQ
  • Roadmap
  • Changelog

Packages

npm

Stay in touch

Get open source security insights delivered straight into your inbox.


  • Terms
  • Privacy
  • Security

Made with ⚡️ by Socket Inc