-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathindex.js
104 lines (91 loc) · 2.63 KB
/
index.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
"use strict";
const unique = (value, index, self) => {
return self.indexOf(value) === index;
};
const addSideeffect = (t, path, source, insert) => {
const statement = t.importDeclaration([], t.stringLiteral(source));
if (insert === "after") path.insertAfter(statement);
else path.insertBefore(statement);
};
const isMatch = (pattern, source) => new RegExp(pattern).test(source);
const makeCompile =
({ source, scope, package: pkgName, specifier }) =>
(mapping) => {
if (typeof mapping === "string")
return mapping
.replace("[*]", source)
.replace("[scope]", scope)
.replace("[package]", pkgName)
.replace("[specifier]", specifier);
if (typeof mapping === "function")
return mapping({
source,
scope,
package: pkgName,
specifier,
});
throw (
"Type of " +
typeof mapping +
"is not supported as a mapping value. Must be either string, function or an array of those."
);
};
const makeApplySideeffect =
(t, path, specifiers, source, sideEffects, scope, pkgName, insert) =>
(pattern) => {
if (!isMatch(pattern, source)) return;
specifiers
.reduce((prev, current) => {
const specifierName = current.local.name;
const compile = makeCompile({
source,
scope,
package: pkgName,
specifier: specifierName,
});
const mapping = sideEffects[pattern];
return [
...prev,
...(Array.isArray(mapping)
? mapping.map(compile)
: [compile(mapping)]),
];
return prev;
}, [])
.filter(Boolean)
.filter(unique)
.forEach((mapping) => {
addSideeffect(t, path, mapping, insert);
});
};
const plugin = ({ types: t }) => {
return {
visitor: {
ImportDeclaration(
path,
{ opts: { sideEffects = {}, insert = "before" } = {} } = {}
) {
if (path.node.specifiers.length === 0)
/* skip sideeffect imports to prevent infinite loops */ return;
const source = path.node.source.value;
const scope = source.startsWith("@") ? source.split("/")[0] : "";
const pkgName = source.startsWith("@")
? source.split("/")[1]
: source.split("/")[0];
const specifiers = path.node.specifiers;
const applySideeffect = makeApplySideeffect(
t,
path,
specifiers,
source,
sideEffects,
scope,
pkgName,
insert
);
Object.keys(sideEffects).forEach(applySideeffect);
},
},
};
};
exports.default = plugin;