1 | ;
|
2 |
|
3 | var _declaredScope = require('eslint-module-utils/declaredScope');
|
4 |
|
5 | var _declaredScope2 = _interopRequireDefault(_declaredScope);
|
6 |
|
7 | var _ExportMap = require('../ExportMap');
|
8 |
|
9 | var _ExportMap2 = _interopRequireDefault(_ExportMap);
|
10 |
|
11 | var _importDeclaration = require('../importDeclaration');
|
12 |
|
13 | var _importDeclaration2 = _interopRequireDefault(_importDeclaration);
|
14 |
|
15 | var _docsUrl = require('../docsUrl');
|
16 |
|
17 | var _docsUrl2 = _interopRequireDefault(_docsUrl);
|
18 |
|
19 | function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
|
20 |
|
21 | module.exports = {
|
22 | meta: {
|
23 | docs: {
|
24 | url: (0, _docsUrl2.default)('namespace')
|
25 | },
|
26 |
|
27 | schema: [{
|
28 | 'type': 'object',
|
29 | 'properties': {
|
30 | 'allowComputed': {
|
31 | 'description': 'If `false`, will report computed (and thus, un-lintable) references ' + 'to namespace members.',
|
32 | 'type': 'boolean',
|
33 | 'default': false
|
34 | }
|
35 | },
|
36 | 'additionalProperties': false
|
37 | }]
|
38 | },
|
39 |
|
40 | create: function namespaceRule(context) {
|
41 |
|
42 | // read options
|
43 | var _ref = context.options[0] || {},
|
44 | _ref$allowComputed = _ref.allowComputed;
|
45 |
|
46 | const allowComputed = _ref$allowComputed === undefined ? false : _ref$allowComputed;
|
47 |
|
48 |
|
49 | const namespaces = new Map();
|
50 |
|
51 | function makeMessage(last, namepath) {
|
52 | return `'${last.name}' not found in` + (namepath.length > 1 ? ' deeply ' : ' ') + `imported namespace '${namepath.join('.')}'.`;
|
53 | }
|
54 |
|
55 | return {
|
56 |
|
57 | // pick up all imports at body entry time, to properly respect hoisting
|
58 | Program: function (_ref2) {
|
59 | let body = _ref2.body;
|
60 |
|
61 | function processBodyStatement(declaration) {
|
62 | if (declaration.type !== 'ImportDeclaration') return;
|
63 |
|
64 | if (declaration.specifiers.length === 0) return;
|
65 |
|
66 | const imports = _ExportMap2.default.get(declaration.source.value, context);
|
67 | if (imports == null) return null;
|
68 |
|
69 | if (imports.errors.length) {
|
70 | imports.reportErrors(context, declaration);
|
71 | return;
|
72 | }
|
73 |
|
74 | for (const specifier of declaration.specifiers) {
|
75 | switch (specifier.type) {
|
76 | case 'ImportNamespaceSpecifier':
|
77 | if (!imports.size) {
|
78 | context.report(specifier, `No exported names found in module '${declaration.source.value}'.`);
|
79 | }
|
80 | namespaces.set(specifier.local.name, imports);
|
81 | break;
|
82 | case 'ImportDefaultSpecifier':
|
83 | case 'ImportSpecifier':
|
84 | {
|
85 | const meta = imports.get(
|
86 | // default to 'default' for default http://i.imgur.com/nj6qAWy.jpg
|
87 | specifier.imported ? specifier.imported.name : 'default');
|
88 | if (!meta || !meta.namespace) break;
|
89 | namespaces.set(specifier.local.name, meta.namespace);
|
90 | break;
|
91 | }
|
92 | }
|
93 | }
|
94 | }
|
95 | body.forEach(processBodyStatement);
|
96 | },
|
97 |
|
98 | // same as above, but does not add names to local map
|
99 | ExportNamespaceSpecifier: function (namespace) {
|
100 | var declaration = (0, _importDeclaration2.default)(context);
|
101 |
|
102 | var imports = _ExportMap2.default.get(declaration.source.value, context);
|
103 | if (imports == null) return null;
|
104 |
|
105 | if (imports.errors.length) {
|
106 | imports.reportErrors(context, declaration);
|
107 | return;
|
108 | }
|
109 |
|
110 | if (!imports.size) {
|
111 | context.report(namespace, `No exported names found in module '${declaration.source.value}'.`);
|
112 | }
|
113 | },
|
114 |
|
115 | // todo: check for possible redefinition
|
116 |
|
117 | MemberExpression: function (dereference) {
|
118 | if (dereference.object.type !== 'Identifier') return;
|
119 | if (!namespaces.has(dereference.object.name)) return;
|
120 |
|
121 | if (dereference.parent.type === 'AssignmentExpression' && dereference.parent.left === dereference) {
|
122 | context.report(dereference.parent, `Assignment to member of namespace '${dereference.object.name}'.`);
|
123 | }
|
124 |
|
125 | // go deep
|
126 | var namespace = namespaces.get(dereference.object.name);
|
127 | var namepath = [dereference.object.name];
|
128 | // while property is namespace and parent is member expression, keep validating
|
129 | while (namespace instanceof _ExportMap2.default && dereference.type === 'MemberExpression') {
|
130 |
|
131 | if (dereference.computed) {
|
132 | if (!allowComputed) {
|
133 | context.report(dereference.property, 'Unable to validate computed reference to imported namespace \'' + dereference.object.name + '\'.');
|
134 | }
|
135 | return;
|
136 | }
|
137 |
|
138 | if (!namespace.has(dereference.property.name)) {
|
139 | context.report(dereference.property, makeMessage(dereference.property, namepath));
|
140 | break;
|
141 | }
|
142 |
|
143 | const exported = namespace.get(dereference.property.name);
|
144 | if (exported == null) return;
|
145 |
|
146 | // stash and pop
|
147 | namepath.push(dereference.property.name);
|
148 | namespace = exported.namespace;
|
149 | dereference = dereference.parent;
|
150 | }
|
151 | },
|
152 |
|
153 | VariableDeclarator: function (_ref3) {
|
154 | let id = _ref3.id,
|
155 | init = _ref3.init;
|
156 |
|
157 | if (init == null) return;
|
158 | if (init.type !== 'Identifier') return;
|
159 | if (!namespaces.has(init.name)) return;
|
160 |
|
161 | // check for redefinition in intermediate scopes
|
162 | if ((0, _declaredScope2.default)(context, init.name) !== 'module') return;
|
163 |
|
164 | // DFS traverse child namespaces
|
165 | function testKey(pattern, namespace) {
|
166 | let path = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : [init.name];
|
167 |
|
168 | if (!(namespace instanceof _ExportMap2.default)) return;
|
169 |
|
170 | if (pattern.type !== 'ObjectPattern') return;
|
171 |
|
172 | for (const property of pattern.properties) {
|
173 | if (property.type === 'ExperimentalRestProperty' || property.type === 'RestElement' || !property.key) {
|
174 | continue;
|
175 | }
|
176 |
|
177 | if (property.key.type !== 'Identifier') {
|
178 | context.report({
|
179 | node: property,
|
180 | message: 'Only destructure top-level names.'
|
181 | });
|
182 | continue;
|
183 | }
|
184 |
|
185 | if (!namespace.has(property.key.name)) {
|
186 | context.report({
|
187 | node: property,
|
188 | message: makeMessage(property.key, path)
|
189 | });
|
190 | continue;
|
191 | }
|
192 |
|
193 | path.push(property.key.name);
|
194 | testKey(property.value, namespace.get(property.key.name).namespace, path);
|
195 | path.pop();
|
196 | }
|
197 | }
|
198 |
|
199 | testKey(id, namespaces.get(init.name));
|
200 | },
|
201 |
|
202 | JSXMemberExpression: function (_ref4) {
|
203 | let object = _ref4.object,
|
204 | property = _ref4.property;
|
205 |
|
206 | if (!namespaces.has(object.name)) return;
|
207 | var namespace = namespaces.get(object.name);
|
208 | if (!namespace.has(property.name)) {
|
209 | context.report({
|
210 | node: property,
|
211 | message: makeMessage(property, [object.name])
|
212 | });
|
213 | }
|
214 | }
|
215 | };
|
216 | }
|
217 | };
|
218 | //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["rules/namespace.js"],"names":["module","exports","meta","docs","url","schema","create","namespaceRule","context","options","allowComputed","namespaces","Map","makeMessage","last","namepath","name","length","join","Program","body","processBodyStatement","declaration","type","specifiers","imports","Exports","get","source","value","errors","reportErrors","specifier","size","report","set","local","imported","namespace","forEach","ExportNamespaceSpecifier","MemberExpression","dereference","object","has","parent","left","computed","property","exported","push","VariableDeclarator","id","init","testKey","pattern","path","properties","key","node","message","pop","JSXMemberExpression"],"mappings":"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","file":"rules/namespace.js","sourcesContent":["import declaredScope from 'eslint-module-utils/declaredScope'\nimport Exports from '../ExportMap'\nimport importDeclaration from '../importDeclaration'\nimport docsUrl from '../docsUrl'\n\nmodule.exports = {\n  meta: {\n    docs: {\n      url: docsUrl('namespace'),\n    },\n\n    schema: [\n      {\n        'type': 'object',\n        'properties': {\n          'allowComputed': {\n            'description':\n              'If `false`, will report computed (and thus, un-lintable) references ' +\n              'to namespace members.',\n            'type': 'boolean',\n            'default': false,\n          },\n        },\n        'additionalProperties': false,\n      },\n    ],\n  },\n\n  create: function namespaceRule(context) {\n\n    // read options\n    const {\n      allowComputed = false,\n    } = context.options[0] || {}\n\n    const namespaces = new Map()\n\n    function makeMessage(last, namepath) {\n       return `'${last.name}' not found in` +\n              (namepath.length > 1 ? ' deeply ' : ' ') +\n              `imported namespace '${namepath.join('.')}'.`\n    }\n\n    return {\n\n      // pick up all imports at body entry time, to properly respect hoisting\n      Program: function ({ body }) {\n        function processBodyStatement(declaration) {\n          if (declaration.type !== 'ImportDeclaration') return\n\n          if (declaration.specifiers.length === 0) return\n\n          const imports = Exports.get(declaration.source.value, context)\n          if (imports == null) return null\n\n          if (imports.errors.length) {\n            imports.reportErrors(context, declaration)\n            return\n          }\n\n          for (const specifier of declaration.specifiers) {\n            switch (specifier.type) {\n              case 'ImportNamespaceSpecifier':\n                if (!imports.size) {\n                  context.report(specifier,\n                    `No exported names found in module '${declaration.source.value}'.`)\n                }\n                namespaces.set(specifier.local.name, imports)\n                break\n              case 'ImportDefaultSpecifier':\n              case 'ImportSpecifier': {\n                const meta = imports.get(\n                  // default to 'default' for default http://i.imgur.com/nj6qAWy.jpg\n                  specifier.imported ? specifier.imported.name : 'default')\n                if (!meta || !meta.namespace) break\n                namespaces.set(specifier.local.name, meta.namespace)\n                break\n              }\n            }\n          }\n        }\n        body.forEach(processBodyStatement)\n      },\n\n      // same as above, but does not add names to local map\n      ExportNamespaceSpecifier: function (namespace) {\n        var declaration = importDeclaration(context)\n\n        var imports = Exports.get(declaration.source.value, context)\n        if (imports == null) return null\n\n        if (imports.errors.length) {\n          imports.reportErrors(context, declaration)\n          return\n        }\n\n        if (!imports.size) {\n          context.report(namespace,\n            `No exported names found in module '${declaration.source.value}'.`)\n        }\n      },\n\n      // todo: check for possible redefinition\n\n      MemberExpression: function (dereference) {\n        if (dereference.object.type !== 'Identifier') return\n        if (!namespaces.has(dereference.object.name)) return\n\n        if (dereference.parent.type === 'AssignmentExpression' &&\n            dereference.parent.left === dereference) {\n            context.report(dereference.parent,\n                `Assignment to member of namespace '${dereference.object.name}'.`)\n        }\n\n        // go deep\n        var namespace = namespaces.get(dereference.object.name)\n        var namepath = [dereference.object.name]\n        // while property is namespace and parent is member expression, keep validating\n        while (namespace instanceof Exports &&\n               dereference.type === 'MemberExpression') {\n\n          if (dereference.computed) {\n            if (!allowComputed) {\n              context.report(dereference.property,\n                'Unable to validate computed reference to imported namespace \\'' +\n                dereference.object.name + '\\'.')\n            }\n            return\n          }\n\n          if (!namespace.has(dereference.property.name)) {\n            context.report(\n              dereference.property,\n              makeMessage(dereference.property, namepath))\n            break\n          }\n\n          const exported = namespace.get(dereference.property.name)\n          if (exported == null) return\n\n          // stash and pop\n          namepath.push(dereference.property.name)\n          namespace = exported.namespace\n          dereference = dereference.parent\n        }\n\n      },\n\n      VariableDeclarator: function ({ id, init }) {\n        if (init == null) return\n        if (init.type !== 'Identifier') return\n        if (!namespaces.has(init.name)) return\n\n        // check for redefinition in intermediate scopes\n        if (declaredScope(context, init.name) !== 'module') return\n\n        // DFS traverse child namespaces\n        function testKey(pattern, namespace, path = [init.name]) {\n          if (!(namespace instanceof Exports)) return\n\n          if (pattern.type !== 'ObjectPattern') return\n\n          for (const property of pattern.properties) {\n            if (\n              property.type === 'ExperimentalRestProperty'\n              || property.type === 'RestElement'\n              || !property.key\n            ) {\n              continue\n            }\n\n            if (property.key.type !== 'Identifier') {\n              context.report({\n                node: property,\n                message: 'Only destructure top-level names.',\n              })\n              continue\n            }\n\n            if (!namespace.has(property.key.name)) {\n              context.report({\n                node: property,\n                message: makeMessage(property.key, path),\n              })\n              continue\n            }\n\n            path.push(property.key.name)\n            testKey(property.value, namespace.get(property.key.name).namespace, path)\n            path.pop()\n          }\n        }\n\n        testKey(id, namespaces.get(init.name))\n      },\n\n      JSXMemberExpression: function({object, property}) {\n         if (!namespaces.has(object.name)) return\n         var namespace = namespaces.get(object.name)\n         if (!namespace.has(property.name)) {\n           context.report({\n             node: property,\n             message: makeMessage(property, [object.name]),\n           })\n         }\n      },\n    }\n  },\n}\n"]} |
\ | No newline at end of file |