123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197 |
- import { createFilter, attachScopes, makeLegalIdentifier } from 'rollup-pluginutils';
- import { sep } from 'path';
- import { walk } from 'estree-walker';
- import MagicString from 'magic-string';
- const escape = str => {
- return str.replace(/[-[\]/{}()*+?.\\^$|]/g, "\\$&");
- };
- const isReference = (node, parent) => {
- if (node.type === "MemberExpression") {
- return !node.computed && isReference(node.object, node);
- }
- if (node.type === "Identifier") {
- // TODO is this right?
- if (parent.type === "MemberExpression") return parent.computed || node === parent.object;
- // disregard the `bar` in { bar: foo }
- if (parent.type === "Property" && node !== parent.value) return false;
- // disregard the `bar` in `class Foo { bar () {...} }`
- if (parent.type === "MethodDefinition") return false;
- // disregard the `bar` in `export { foo as bar }`
- if (parent.type === "ExportSpecifier" && node !== parent.local) return;
- return true;
- }
- };
- const flatten = node => {
- const parts = [];
- while (node.type === "MemberExpression") {
- parts.unshift(node.property.name);
- node = node.object;
- }
- const name = node.name;
- parts.unshift(name);
- return { name, keypath: parts.join(".") };
- };
- function inject(options) {
- if (!options) throw new Error("Missing options");
- const filter = createFilter(options.include, options.exclude);
- let modules = options.modules;
- if (!modules) {
- modules = Object.assign({}, options);
- delete modules.include;
- delete modules.exclude;
- delete modules.sourceMap;
- delete modules.sourcemap;
- }
- const modulesMap = new Map(Object.entries(modules));
- // Fix paths on Windows
- if (sep !== "/") {
- modulesMap.forEach((mod, key) => {
- modulesMap.set(
- key,
- Array.isArray(mod) ? [mod[0].split(sep).join("/"), mod[1]] : mod.split(sep).join("/")
- );
- });
- }
- const firstpass = new RegExp(
- `(?:${Array.from(modulesMap.keys())
- .map(escape)
- .join("|")})`,
- "g"
- );
- const sourceMap = options.sourceMap !== false && options.sourcemap !== false;
- return {
- name: "inject",
- transform(code, id) {
- if (!filter(id)) return null;
- if (code.search(firstpass) === -1) return null;
- if (sep !== "/") id = id.split(sep).join("/");
- let ast = null;
- try {
- ast = this.parse(code);
- } catch (err) {
- this.warn({
- code: "PARSE_ERROR",
- message: `rollup-plugin-inject: failed to parse ${id}. Consider restricting the plugin to particular files via options.include`
- });
- }
- if (!ast) {
- return null;
- }
- // analyse scopes
- let scope = attachScopes(ast, "scope");
- const imports = new Set();
- ast.body.forEach(node => {
- if (node.type === "ImportDeclaration") {
- node.specifiers.forEach(specifier => {
- imports.add(specifier.local.name);
- });
- }
- });
- const magicString = new MagicString(code);
- const newImports = new Map();
- function handleReference(node, name, keypath) {
- let mod = modulesMap.get(keypath);
- if (mod && !imports.has(name) && !scope.contains(name)) {
- if (typeof mod === "string") mod = [mod, "default"];
- // prevent module from importing itself
- if (mod[0] === id) return;
- const hash = `${keypath}:${mod[0]}:${mod[1]}`;
- const importLocalName =
- name === keypath ? name : makeLegalIdentifier(`$inject_${keypath}`);
- if (!newImports.has(hash)) {
- if (mod[1] === "*") {
- newImports.set(hash, `import * as ${importLocalName} from '${mod[0]}';`);
- } else {
- newImports.set(hash, `import { ${mod[1]} as ${importLocalName} } from '${mod[0]}';`);
- }
- }
- if (name !== keypath) {
- magicString.overwrite(node.start, node.end, importLocalName, {
- storeName: true
- });
- }
- return true;
- }
- }
- walk(ast, {
- enter(node, parent) {
- if (sourceMap) {
- magicString.addSourcemapLocation(node.start);
- magicString.addSourcemapLocation(node.end);
- }
- if (node.scope) scope = node.scope;
- // special case – shorthand properties. because node.key === node.value,
- // we can't differentiate once we've descended into the node
- if (node.type === "Property" && node.shorthand) {
- const name = node.key.name;
- handleReference(node, name, name);
- return this.skip();
- }
- if (isReference(node, parent)) {
- const { name, keypath } = flatten(node);
- const handled = handleReference(node, name, keypath);
- if (handled) return this.skip();
- }
- },
- leave(node) {
- if (node.scope) scope = scope.parent;
- }
- });
- if (newImports.size === 0) {
- return {
- code,
- ast,
- map: sourceMap ? magicString.generateMap({ hires: true }) : null
- };
- }
- const importBlock = Array.from(newImports.values()).join("\n\n");
- magicString.prepend(importBlock + "\n\n");
- return {
- code: magicString.toString(),
- map: sourceMap ? magicString.generateMap({ hires: true }) : null
- };
- }
- };
- }
- export default inject;
|