219 lines
26 KiB
JavaScript
219 lines
26 KiB
JavaScript
'use strict';var _declaredScope = require('eslint-module-utils/declaredScope');var _declaredScope2 = _interopRequireDefault(_declaredScope);
|
|
var _builder = require('../exportMap/builder');var _builder2 = _interopRequireDefault(_builder);
|
|
var _exportMap = require('../exportMap');var _exportMap2 = _interopRequireDefault(_exportMap);
|
|
var _importDeclaration = require('../importDeclaration');var _importDeclaration2 = _interopRequireDefault(_importDeclaration);
|
|
var _docsUrl = require('../docsUrl');var _docsUrl2 = _interopRequireDefault(_docsUrl);function _interopRequireDefault(obj) {return obj && obj.__esModule ? obj : { 'default': obj };}
|
|
|
|
function processBodyStatement(context, namespaces, declaration) {
|
|
if (declaration.type !== 'ImportDeclaration') {return;}
|
|
|
|
if (declaration.specifiers.length === 0) {return;}
|
|
|
|
var imports = _builder2['default'].get(declaration.source.value, context);
|
|
if (imports == null) {return null;}
|
|
|
|
if (imports.errors.length > 0) {
|
|
imports.reportErrors(context, declaration);
|
|
return;
|
|
}
|
|
|
|
declaration.specifiers.forEach(function (specifier) {
|
|
switch (specifier.type) {
|
|
case 'ImportNamespaceSpecifier':
|
|
if (!imports.size) {
|
|
context.report(
|
|
specifier, 'No exported names found in module \'' + String(
|
|
declaration.source.value) + '\'.');
|
|
|
|
}
|
|
namespaces.set(specifier.local.name, imports);
|
|
break;
|
|
case 'ImportDefaultSpecifier':
|
|
case 'ImportSpecifier':{
|
|
var meta = imports.get(
|
|
// default to 'default' for default https://i.imgur.com/nj6qAWy.jpg
|
|
specifier.imported ? specifier.imported.name || specifier.imported.value : 'default');
|
|
|
|
if (!meta || !meta.namespace) {break;}
|
|
namespaces.set(specifier.local.name, meta.namespace);
|
|
break;
|
|
}
|
|
default:}
|
|
|
|
});
|
|
}
|
|
|
|
module.exports = {
|
|
meta: {
|
|
type: 'problem',
|
|
docs: {
|
|
category: 'Static analysis',
|
|
description: 'Ensure imported namespaces contain dereferenced properties as they are dereferenced.',
|
|
url: (0, _docsUrl2['default'])('namespace') },
|
|
|
|
|
|
schema: [
|
|
{
|
|
type: 'object',
|
|
properties: {
|
|
allowComputed: {
|
|
description: 'If `false`, will report computed (and thus, un-lintable) references to namespace members.',
|
|
type: 'boolean',
|
|
'default': false } },
|
|
|
|
|
|
additionalProperties: false }] },
|
|
|
|
|
|
|
|
|
|
create: function () {function namespaceRule(context) {
|
|
// read options
|
|
var _ref =
|
|
|
|
context.options[0] || {},_ref$allowComputed = _ref.allowComputed,allowComputed = _ref$allowComputed === undefined ? false : _ref$allowComputed;
|
|
|
|
var namespaces = new Map();
|
|
|
|
function makeMessage(last, namepath) {
|
|
return '\'' + String(last.name) + '\' not found in ' + (namepath.length > 1 ? 'deeply ' : '') + 'imported namespace \'' + String(namepath.join('.')) + '\'.';
|
|
}
|
|
|
|
return {
|
|
// pick up all imports at body entry time, to properly respect hoisting
|
|
Program: function () {function Program(_ref2) {var body = _ref2.body;
|
|
body.forEach(function (x) {processBodyStatement(context, namespaces, x);});
|
|
}return Program;}(),
|
|
|
|
// same as above, but does not add names to local map
|
|
ExportNamespaceSpecifier: function () {function ExportNamespaceSpecifier(namespace) {
|
|
var declaration = (0, _importDeclaration2['default'])(context);
|
|
|
|
var imports = _builder2['default'].get(declaration.source.value, context);
|
|
if (imports == null) {return null;}
|
|
|
|
if (imports.errors.length) {
|
|
imports.reportErrors(context, declaration);
|
|
return;
|
|
}
|
|
|
|
if (!imports.size) {
|
|
context.report(
|
|
namespace, 'No exported names found in module \'' + String(
|
|
declaration.source.value) + '\'.');
|
|
|
|
}
|
|
}return ExportNamespaceSpecifier;}(),
|
|
|
|
// todo: check for possible redefinition
|
|
|
|
MemberExpression: function () {function MemberExpression(dereference) {
|
|
if (dereference.object.type !== 'Identifier') {return;}
|
|
if (!namespaces.has(dereference.object.name)) {return;}
|
|
if ((0, _declaredScope2['default'])(context, dereference.object.name) !== 'module') {return;}
|
|
|
|
if (dereference.parent.type === 'AssignmentExpression' && dereference.parent.left === dereference) {
|
|
context.report(
|
|
dereference.parent, 'Assignment to member of namespace \'' + String(
|
|
dereference.object.name) + '\'.');
|
|
|
|
}
|
|
|
|
// go deep
|
|
var namespace = namespaces.get(dereference.object.name);
|
|
var namepath = [dereference.object.name];
|
|
// while property is namespace and parent is member expression, keep validating
|
|
while (namespace instanceof _exportMap2['default'] && dereference.type === 'MemberExpression') {
|
|
if (dereference.computed) {
|
|
if (!allowComputed) {
|
|
context.report(
|
|
dereference.property, 'Unable to validate computed reference to imported namespace \'' + String(
|
|
dereference.object.name) + '\'.');
|
|
|
|
}
|
|
return;
|
|
}
|
|
|
|
if (!namespace.has(dereference.property.name)) {
|
|
context.report(
|
|
dereference.property,
|
|
makeMessage(dereference.property, namepath));
|
|
|
|
break;
|
|
}
|
|
|
|
var exported = namespace.get(dereference.property.name);
|
|
if (exported == null) {return;}
|
|
|
|
// stash and pop
|
|
namepath.push(dereference.property.name);
|
|
namespace = exported.namespace;
|
|
dereference = dereference.parent;
|
|
}
|
|
}return MemberExpression;}(),
|
|
|
|
VariableDeclarator: function () {function VariableDeclarator(_ref3) {var id = _ref3.id,init = _ref3.init;
|
|
if (init == null) {return;}
|
|
if (init.type !== 'Identifier') {return;}
|
|
if (!namespaces.has(init.name)) {return;}
|
|
|
|
// check for redefinition in intermediate scopes
|
|
if ((0, _declaredScope2['default'])(context, init.name) !== 'module') {return;}
|
|
|
|
// DFS traverse child namespaces
|
|
function testKey(pattern, namespace) {var path = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : [init.name];
|
|
if (!(namespace instanceof _exportMap2['default'])) {return;}
|
|
|
|
if (pattern.type !== 'ObjectPattern') {return;}var _iteratorNormalCompletion = true;var _didIteratorError = false;var _iteratorError = undefined;try {
|
|
|
|
for (var _iterator = pattern.properties[Symbol.iterator](), _step; !(_iteratorNormalCompletion = (_step = _iterator.next()).done); _iteratorNormalCompletion = true) {var property = _step.value;
|
|
if (
|
|
property.type === 'ExperimentalRestProperty' ||
|
|
property.type === 'RestElement' ||
|
|
!property.key)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
if (property.key.type !== 'Identifier') {
|
|
context.report({
|
|
node: property,
|
|
message: 'Only destructure top-level names.' });
|
|
|
|
continue;
|
|
}
|
|
|
|
if (!namespace.has(property.key.name)) {
|
|
context.report({
|
|
node: property,
|
|
message: makeMessage(property.key, path) });
|
|
|
|
continue;
|
|
}
|
|
|
|
path.push(property.key.name);
|
|
var dependencyExportMap = namespace.get(property.key.name);
|
|
// could be null when ignored or ambiguous
|
|
if (dependencyExportMap !== null) {
|
|
testKey(property.value, dependencyExportMap.namespace, path);
|
|
}
|
|
path.pop();
|
|
}} catch (err) {_didIteratorError = true;_iteratorError = err;} finally {try {if (!_iteratorNormalCompletion && _iterator['return']) {_iterator['return']();}} finally {if (_didIteratorError) {throw _iteratorError;}}}
|
|
}
|
|
|
|
testKey(id, namespaces.get(init.name));
|
|
}return VariableDeclarator;}(),
|
|
|
|
JSXMemberExpression: function () {function JSXMemberExpression(_ref4) {var object = _ref4.object,property = _ref4.property;
|
|
if (!namespaces.has(object.name)) {return;}
|
|
var namespace = namespaces.get(object.name);
|
|
if (!namespace.has(property.name)) {
|
|
context.report({
|
|
node: property,
|
|
message: makeMessage(property, [object.name]) });
|
|
|
|
}
|
|
}return JSXMemberExpression;}() };
|
|
|
|
}return namespaceRule;}() };
|
|
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/rules/namespace.js"],"names":["processBodyStatement","context","namespaces","declaration","type","specifiers","length","imports","ExportMapBuilder","get","source","value","errors","reportErrors","forEach","specifier","size","report","set","local","name","meta","imported","namespace","module","exports","docs","category","description","url","schema","properties","allowComputed","additionalProperties","create","namespaceRule","options","Map","makeMessage","last","namepath","join","Program","body","x","ExportNamespaceSpecifier","MemberExpression","dereference","object","has","parent","left","ExportMap","computed","property","exported","push","VariableDeclarator","id","init","testKey","pattern","path","key","node","message","dependencyExportMap","pop","JSXMemberExpression"],"mappings":"aAAA,kE;AACA,+C;AACA,yC;AACA,yD;AACA,qC;;AAEA,SAASA,oBAAT,CAA8BC,OAA9B,EAAuCC,UAAvC,EAAmDC,WAAnD,EAAgE;AAC9D,MAAIA,YAAYC,IAAZ,KAAqB,mBAAzB,EAA8C,CAAE,OAAS;;AAEzD,MAAID,YAAYE,UAAZ,CAAuBC,MAAvB,KAAkC,CAAtC,EAAyC,CAAE,OAAS;;AAEpD,MAAMC,UAAUC,qBAAiBC,GAAjB,CAAqBN,YAAYO,MAAZ,CAAmBC,KAAxC,EAA+CV,OAA/C,CAAhB;AACA,MAAIM,WAAW,IAAf,EAAqB,CAAE,OAAO,IAAP,CAAc;;AAErC,MAAIA,QAAQK,MAAR,CAAeN,MAAf,GAAwB,CAA5B,EAA+B;AAC7BC,YAAQM,YAAR,CAAqBZ,OAArB,EAA8BE,WAA9B;AACA;AACD;;AAEDA,cAAYE,UAAZ,CAAuBS,OAAvB,CAA+B,UAACC,SAAD,EAAe;AAC5C,YAAQA,UAAUX,IAAlB;AACE,WAAK,0BAAL;AACE,YAAI,CAACG,QAAQS,IAAb,EAAmB;AACjBf,kBAAQgB,MAAR;AACEF,mBADF;AAEwCZ,sBAAYO,MAAZ,CAAmBC,KAF3D;;AAID;AACDT,mBAAWgB,GAAX,CAAeH,UAAUI,KAAV,CAAgBC,IAA/B,EAAqCb,OAArC;AACA;AACF,WAAK,wBAAL;AACA,WAAK,iBAAL,CAAwB;AACtB,cAAMc,OAAOd,QAAQE,GAAR;AACb;AACEM,oBAAUO,QAAV,GAAqBP,UAAUO,QAAV,CAAmBF,IAAnB,IAA2BL,UAAUO,QAAV,CAAmBX,KAAnE,GAA2E,SAFhE,CAAb;;AAIA,cAAI,CAACU,IAAD,IAAS,CAACA,KAAKE,SAAnB,EAA8B,CAAE,MAAQ;AACxCrB,qBAAWgB,GAAX,CAAeH,UAAUI,KAAV,CAAgBC,IAA/B,EAAqCC,KAAKE,SAA1C;AACA;AACD;AACD,cApBF;;AAsBD,GAvBD;AAwBD;;AAEDC,OAAOC,OAAP,GAAiB;AACfJ,QAAM;AACJjB,UAAM,SADF;AAEJsB,UAAM;AACJC,gBAAU,iBADN;AAEJC,mBAAa,sFAFT;AAGJC,WAAK,0BAAQ,WAAR,CAHD,EAFF;;;AAQJC,YAAQ;AACN;AACE1B,YAAM,QADR;AAEE2B,kBAAY;AACVC,uBAAe;AACbJ,uBAAa,2FADA;AAEbxB,gBAAM,SAFO;AAGb,qBAAS,KAHI,EADL,EAFd;;;AASE6B,4BAAsB,KATxB,EADM,CARJ,EADS;;;;;AAwBfC,uBAAQ,SAASC,aAAT,CAAuBlC,OAAvB,EAAgC;AACtC;AADsC;;AAIlCA,cAAQmC,OAAR,CAAgB,CAAhB,KAAsB,EAJY,2BAGpCJ,aAHoC,CAGpCA,aAHoC,sCAGpB,KAHoB;;AAMtC,UAAM9B,aAAa,IAAImC,GAAJ,EAAnB;;AAEA,eAASC,WAAT,CAAqBC,IAArB,EAA2BC,QAA3B,EAAqC;AACnC,6BAAWD,KAAKnB,IAAhB,0BAAsCoB,SAASlC,MAAT,GAAkB,CAAlB,GAAsB,SAAtB,GAAkC,EAAxE,qCAAiGkC,SAASC,IAAT,CAAc,GAAd,CAAjG;AACD;;AAED,aAAO;AACL;AACAC,eAFK,uCAEa,KAARC,IAAQ,SAARA,IAAQ;AAChBA,iBAAK7B,OAAL,CAAa,UAAC8B,CAAD,EAAO,CAAE5C,qBAAqBC,OAArB,EAA8BC,UAA9B,EAA0C0C,CAA1C,EAA+C,CAArE;AACD,WAJI;;AAML;AACAC,gCAPK,iDAOoBtB,SAPpB,EAO+B;AAClC,gBAAMpB,cAAc,oCAAkBF,OAAlB,CAApB;;AAEA,gBAAMM,UAAUC,qBAAiBC,GAAjB,CAAqBN,YAAYO,MAAZ,CAAmBC,KAAxC,EAA+CV,OAA/C,CAAhB;AACA,gBAAIM,WAAW,IAAf,EAAqB,CAAE,OAAO,IAAP,CAAc;;AAErC,gBAAIA,QAAQK,MAAR,CAAeN,MAAnB,EAA2B;AACzBC,sBAAQM,YAAR,CAAqBZ,OAArB,EAA8BE,WAA9B;AACA;AACD;;AAED,gBAAI,CAACI,QAAQS,IAAb,EAAmB;AACjBf,sBAAQgB,MAAR;AACEM,uBADF;AAEwCpB,0BAAYO,MAAZ,CAAmBC,KAF3D;;AAID;AACF,WAxBI;;AA0BL;;AAEAmC,wBA5BK,yCA4BYC,WA5BZ,EA4ByB;AAC5B,gBAAIA,YAAYC,MAAZ,CAAmB5C,IAAnB,KAA4B,YAAhC,EAA8C,CAAE,OAAS;AACzD,gBAAI,CAACF,WAAW+C,GAAX,CAAeF,YAAYC,MAAZ,CAAmB5B,IAAlC,CAAL,EAA8C,CAAE,OAAS;AACzD,gBAAI,gCAAcnB,OAAd,EAAuB8C,YAAYC,MAAZ,CAAmB5B,IAA1C,MAAoD,QAAxD,EAAkE,CAAE,OAAS;;AAE7E,gBAAI2B,YAAYG,MAAZ,CAAmB9C,IAAnB,KAA4B,sBAA5B,IAAsD2C,YAAYG,MAAZ,CAAmBC,IAAnB,KAA4BJ,WAAtF,EAAmG;AACjG9C,sBAAQgB,MAAR;AACE8B,0BAAYG,MADd;AAEwCH,0BAAYC,MAAZ,CAAmB5B,IAF3D;;AAID;;AAED;AACA,gBAAIG,YAAYrB,WAAWO,GAAX,CAAesC,YAAYC,MAAZ,CAAmB5B,IAAlC,CAAhB;AACA,gBAAMoB,WAAW,CAACO,YAAYC,MAAZ,CAAmB5B,IAApB,CAAjB;AACA;AACA,mBAAOG,qBAAqB6B,sBAArB,IAAkCL,YAAY3C,IAAZ,KAAqB,kBAA9D,EAAkF;AAChF,kBAAI2C,YAAYM,QAAhB,EAA0B;AACxB,oBAAI,CAACrB,aAAL,EAAoB;AAClB/B,0BAAQgB,MAAR;AACE8B,8BAAYO,QADd;AAEkEP,8BAAYC,MAAZ,CAAmB5B,IAFrF;;AAID;AACD;AACD;;AAED,kBAAI,CAACG,UAAU0B,GAAV,CAAcF,YAAYO,QAAZ,CAAqBlC,IAAnC,CAAL,EAA+C;AAC7CnB,wBAAQgB,MAAR;AACE8B,4BAAYO,QADd;AAEEhB,4BAAYS,YAAYO,QAAxB,EAAkCd,QAAlC,CAFF;;AAIA;AACD;;AAED,kBAAMe,WAAWhC,UAAUd,GAAV,CAAcsC,YAAYO,QAAZ,CAAqBlC,IAAnC,CAAjB;AACA,kBAAImC,YAAY,IAAhB,EAAsB,CAAE,OAAS;;AAEjC;AACAf,uBAASgB,IAAT,CAAcT,YAAYO,QAAZ,CAAqBlC,IAAnC;AACAG,0BAAYgC,SAAShC,SAArB;AACAwB,4BAAcA,YAAYG,MAA1B;AACD;AACF,WAvEI;;AAyELO,0BAzEK,kDAyE4B,KAAZC,EAAY,SAAZA,EAAY,CAARC,IAAQ,SAARA,IAAQ;AAC/B,gBAAIA,QAAQ,IAAZ,EAAkB,CAAE,OAAS;AAC7B,gBAAIA,KAAKvD,IAAL,KAAc,YAAlB,EAAgC,CAAE,OAAS;AAC3C,gBAAI,CAACF,WAAW+C,GAAX,CAAeU,KAAKvC,IAApB,CAAL,EAAgC,CAAE,OAAS;;AAE3C;AACA,gBAAI,gCAAcnB,OAAd,EAAuB0D,KAAKvC,IAA5B,MAAsC,QAA1C,EAAoD,CAAE,OAAS;;AAE/D;AACA,qBAASwC,OAAT,CAAiBC,OAAjB,EAA0BtC,SAA1B,EAAyD,KAApBuC,IAAoB,uEAAb,CAACH,KAAKvC,IAAN,CAAa;AACvD,kBAAI,EAAEG,qBAAqB6B,sBAAvB,CAAJ,EAAuC,CAAE,OAAS;;AAElD,kBAAIS,QAAQzD,IAAR,KAAiB,eAArB,EAAsC,CAAE,OAAS,CAHM;;AAKvD,qCAAuByD,QAAQ9B,UAA/B,8HAA2C,KAAhCuB,QAAgC;AACzC;AACEA,2BAASlD,IAAT,KAAkB,0BAAlB;AACGkD,2BAASlD,IAAT,KAAkB,aADrB;AAEG,mBAACkD,SAASS,GAHf;AAIE;AACA;AACD;;AAED,sBAAIT,SAASS,GAAT,CAAa3D,IAAb,KAAsB,YAA1B,EAAwC;AACtCH,4BAAQgB,MAAR,CAAe;AACb+C,4BAAMV,QADO;AAEbW,+BAAS,mCAFI,EAAf;;AAIA;AACD;;AAED,sBAAI,CAAC1C,UAAU0B,GAAV,CAAcK,SAASS,GAAT,CAAa3C,IAA3B,CAAL,EAAuC;AACrCnB,4BAAQgB,MAAR,CAAe;AACb+C,4BAAMV,QADO;AAEbW,+BAAS3B,YAAYgB,SAASS,GAArB,EAA0BD,IAA1B,CAFI,EAAf;;AAIA;AACD;;AAEDA,uBAAKN,IAAL,CAAUF,SAASS,GAAT,CAAa3C,IAAvB;AACA,sBAAM8C,sBAAsB3C,UAAUd,GAAV,CAAc6C,SAASS,GAAT,CAAa3C,IAA3B,CAA5B;AACA;AACA,sBAAI8C,wBAAwB,IAA5B,EAAkC;AAChCN,4BAAQN,SAAS3C,KAAjB,EAAwBuD,oBAAoB3C,SAA5C,EAAuDuC,IAAvD;AACD;AACDA,uBAAKK,GAAL;AACD,iBArCsD;AAsCxD;;AAEDP,oBAAQF,EAAR,EAAYxD,WAAWO,GAAX,CAAekD,KAAKvC,IAApB,CAAZ;AACD,WA3HI;;AA6HLgD,2BA7HK,mDA6HqC,KAApBpB,MAAoB,SAApBA,MAAoB,CAAZM,QAAY,SAAZA,QAAY;AACxC,gBAAI,CAACpD,WAAW+C,GAAX,CAAeD,OAAO5B,IAAtB,CAAL,EAAkC,CAAE,OAAS;AAC7C,gBAAMG,YAAYrB,WAAWO,GAAX,CAAeuC,OAAO5B,IAAtB,CAAlB;AACA,gBAAI,CAACG,UAAU0B,GAAV,CAAcK,SAASlC,IAAvB,CAAL,EAAmC;AACjCnB,sBAAQgB,MAAR,CAAe;AACb+C,sBAAMV,QADO;AAEbW,yBAAS3B,YAAYgB,QAAZ,EAAsB,CAACN,OAAO5B,IAAR,CAAtB,CAFI,EAAf;;AAID;AACF,WAtII,gCAAP;;AAwID,KApJD,OAAiBe,aAAjB,IAxBe,EAAjB","file":"namespace.js","sourcesContent":["import declaredScope from 'eslint-module-utils/declaredScope';\nimport ExportMapBuilder from '../exportMap/builder';\nimport ExportMap from '../exportMap';\nimport importDeclaration from '../importDeclaration';\nimport docsUrl from '../docsUrl';\n\nfunction processBodyStatement(context, namespaces, declaration) {\n  if (declaration.type !== 'ImportDeclaration') { return; }\n\n  if (declaration.specifiers.length === 0) { return; }\n\n  const imports = ExportMapBuilder.get(declaration.source.value, context);\n  if (imports == null) { return null; }\n\n  if (imports.errors.length > 0) {\n    imports.reportErrors(context, declaration);\n    return;\n  }\n\n  declaration.specifiers.forEach((specifier) => {\n    switch (specifier.type) {\n      case 'ImportNamespaceSpecifier':\n        if (!imports.size) {\n          context.report(\n            specifier,\n            `No exported names found in module '${declaration.source.value}'.`,\n          );\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 https://i.imgur.com/nj6qAWy.jpg\n          specifier.imported ? specifier.imported.name || specifier.imported.value : 'default',\n        );\n        if (!meta || !meta.namespace) { break; }\n        namespaces.set(specifier.local.name, meta.namespace);\n        break;\n      }\n      default:\n    }\n  });\n}\n\nmodule.exports = {\n  meta: {\n    type: 'problem',\n    docs: {\n      category: 'Static analysis',\n      description: 'Ensure imported namespaces contain dereferenced properties as they are dereferenced.',\n      url: docsUrl('namespace'),\n    },\n\n    schema: [\n      {\n        type: 'object',\n        properties: {\n          allowComputed: {\n            description: 'If `false`, will report computed (and thus, un-lintable) references to namespace members.',\n            type: 'boolean',\n            default: false,\n          },\n        },\n        additionalProperties: false,\n      },\n    ],\n  },\n\n  create: function namespaceRule(context) {\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 ${namepath.length > 1 ? 'deeply ' : ''}imported namespace '${namepath.join('.')}'.`;\n    }\n\n    return {\n      // pick up all imports at body entry time, to properly respect hoisting\n      Program({ body }) {\n        body.forEach((x) => { processBodyStatement(context, namespaces, x); });\n      },\n\n      // same as above, but does not add names to local map\n      ExportNamespaceSpecifier(namespace) {\n        const declaration = importDeclaration(context);\n\n        const imports = ExportMapBuilder.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(\n            namespace,\n            `No exported names found in module '${declaration.source.value}'.`,\n          );\n        }\n      },\n\n      // todo: check for possible redefinition\n\n      MemberExpression(dereference) {\n        if (dereference.object.type !== 'Identifier') { return; }\n        if (!namespaces.has(dereference.object.name)) { return; }\n        if (declaredScope(context, dereference.object.name) !== 'module') { return; }\n\n        if (dereference.parent.type === 'AssignmentExpression' && dereference.parent.left === dereference) {\n          context.report(\n            dereference.parent,\n            `Assignment to member of namespace '${dereference.object.name}'.`,\n          );\n        }\n\n        // go deep\n        let namespace = namespaces.get(dereference.object.name);\n        const namepath = [dereference.object.name];\n        // while property is namespace and parent is member expression, keep validating\n        while (namespace instanceof ExportMap && dereference.type === 'MemberExpression') {\n          if (dereference.computed) {\n            if (!allowComputed) {\n              context.report(\n                dereference.property,\n                `Unable to validate computed reference to imported namespace '${dereference.object.name}'.`,\n              );\n            }\n            return;\n          }\n\n          if (!namespace.has(dereference.property.name)) {\n            context.report(\n              dereference.property,\n              makeMessage(dereference.property, namepath),\n            );\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      VariableDeclarator({ 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 ExportMap)) { 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            const dependencyExportMap = namespace.get(property.key.name);\n            // could be null when ignored or ambiguous\n            if (dependencyExportMap !== null) {\n              testKey(property.value, dependencyExportMap.namespace, path);\n            }\n            path.pop();\n          }\n        }\n\n        testKey(id, namespaces.get(init.name));\n      },\n\n      JSXMemberExpression({ object, property }) {\n        if (!namespaces.has(object.name)) { return; }\n        const 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"]}
|