'use strict';

var _ExportMap = require('../ExportMap');

var _ExportMap2 = _interopRequireDefault(_ExportMap);

var _resolve = require('eslint-module-utils/resolve');

var _resolve2 = _interopRequireDefault(_resolve);

var _docsUrl = require('../docsUrl');

var _docsUrl2 = _interopRequireDefault(_docsUrl);

function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }

function _toConsumableArray(arr) { if (Array.isArray(arr)) { for (var i = 0, arr2 = Array(arr.length); i < arr.length; i++) arr2[i] = arr[i]; return arr2; } else { return Array.from(arr); } } /**
                                                                                                                                                                                                 * @fileOverview Ensures that modules contain exports and/or all 
                                                                                                                                                                                                 * modules are consumed within other modules.
                                                                                                                                                                                                 * @author René Fermann
                                                                                                                                                                                                 */

// eslint/lib/util/glob-util has been moved to eslint/lib/util/glob-utils with version 5.3
let listFilesToProcess;
try {
  listFilesToProcess = require('eslint/lib/util/glob-utils').listFilesToProcess;
} catch (err) {
  listFilesToProcess = require('eslint/lib/util/glob-util').listFilesToProcess;
}

const EXPORT_DEFAULT_DECLARATION = 'ExportDefaultDeclaration';
const EXPORT_NAMED_DECLARATION = 'ExportNamedDeclaration';
const EXPORT_ALL_DECLARATION = 'ExportAllDeclaration';
const IMPORT_DECLARATION = 'ImportDeclaration';
const IMPORT_NAMESPACE_SPECIFIER = 'ImportNamespaceSpecifier';
const IMPORT_DEFAULT_SPECIFIER = 'ImportDefaultSpecifier';
const VARIABLE_DECLARATION = 'VariableDeclaration';
const FUNCTION_DECLARATION = 'FunctionDeclaration';
const DEFAULT = 'default';

let preparationDone = false;
const importList = new Map();
const exportList = new Map();
const ignoredFiles = new Set();

const isNodeModule = path => {
  return (/\/(node_modules)\//.test(path)
  );
};

/**
 * read all files matching the patterns in src and ignoreExports
 * 
 * return all files matching src pattern, which are not matching the ignoreExports pattern
 */
const resolveFiles = (src, ignoreExports) => {
  const srcFiles = new Set();
  const srcFileList = listFilesToProcess(src);

  // prepare list of ignored files
  const ignoredFilesList = listFilesToProcess(ignoreExports);
  ignoredFilesList.forEach((_ref) => {
    let filename = _ref.filename;
    return ignoredFiles.add(filename);
  });

  // prepare list of source files, don't consider files from node_modules
  srcFileList.filter((_ref2) => {
    let filename = _ref2.filename;
    return !isNodeModule(filename);
  }).forEach((_ref3) => {
    let filename = _ref3.filename;

    srcFiles.add(filename);
  });
  return srcFiles;
};

/**
 * parse all source files and build up 2 maps containing the existing imports and exports
 */
const prepareImportsAndExports = (srcFiles, context) => {
  const exportAll = new Map();
  srcFiles.forEach(file => {
    const exports = new Map();
    const imports = new Map();
    const currentExports = _ExportMap2.default.get(file, context);
    if (currentExports) {
      const dependencies = currentExports.dependencies,
            reexports = currentExports.reexports,
            localImportList = currentExports.imports,
            namespace = currentExports.namespace;

      // dependencies === export * from 

      const currentExportAll = new Set();
      dependencies.forEach(value => {
        currentExportAll.add(value().path);
      });
      exportAll.set(file, currentExportAll);

      reexports.forEach((value, key) => {
        if (key === DEFAULT) {
          exports.set(IMPORT_DEFAULT_SPECIFIER, { whereUsed: new Set() });
        } else {
          exports.set(key, { whereUsed: new Set() });
        }
        const reexport = value.getImport();
        if (!reexport) {
          return;
        }
        let localImport = imports.get(reexport.path);
        let currentValue;
        if (value.local === DEFAULT) {
          currentValue = IMPORT_DEFAULT_SPECIFIER;
        } else {
          currentValue = value.local;
        }
        if (typeof localImport !== 'undefined') {
          localImport = new Set([].concat(_toConsumableArray(localImport), [currentValue]));
        } else {
          localImport = new Set([currentValue]);
        }
        imports.set(reexport.path, localImport);
      });

      localImportList.forEach((value, key) => {
        if (isNodeModule(key)) {
          return;
        }
        imports.set(key, value.importedSpecifiers);
      });
      importList.set(file, imports);

      // build up export list only, if file is not ignored
      if (ignoredFiles.has(file)) {
        return;
      }
      namespace.forEach((value, key) => {
        if (key === DEFAULT) {
          exports.set(IMPORT_DEFAULT_SPECIFIER, { whereUsed: new Set() });
        } else {
          exports.set(key, { whereUsed: new Set() });
        }
      });
    }
    exports.set(EXPORT_ALL_DECLARATION, { whereUsed: new Set() });
    exports.set(IMPORT_NAMESPACE_SPECIFIER, { whereUsed: new Set() });
    exportList.set(file, exports);
  });
  exportAll.forEach((value, key) => {
    value.forEach(val => {
      const currentExports = exportList.get(val);
      const currentExport = currentExports.get(EXPORT_ALL_DECLARATION);
      currentExport.whereUsed.add(key);
    });
  });
};

/**
 * traverse through all imports and add the respective path to the whereUsed-list 
 * of the corresponding export
 */
const determineUsage = () => {
  importList.forEach((listValue, listKey) => {
    listValue.forEach((value, key) => {
      const exports = exportList.get(key);
      if (typeof exports !== 'undefined') {
        value.forEach(currentImport => {
          let specifier;
          if (currentImport === IMPORT_NAMESPACE_SPECIFIER) {
            specifier = IMPORT_NAMESPACE_SPECIFIER;
          } else if (currentImport === IMPORT_DEFAULT_SPECIFIER) {
            specifier = IMPORT_DEFAULT_SPECIFIER;
          } else {
            specifier = currentImport;
          }
          if (typeof specifier !== 'undefined') {
            const exportStatement = exports.get(specifier);
            if (typeof exportStatement !== 'undefined') {
              const whereUsed = exportStatement.whereUsed;

              whereUsed.add(listKey);
              exports.set(specifier, { whereUsed });
            }
          }
        });
      }
    });
  });
};

const getSrc = src => {
  if (src) {
    return src;
  }
  return [process.cwd()];
};

/**
 * prepare the lists of existing imports and exports - should only be executed once at
 * the start of a new eslint run
 */
const doPreparation = (src, ignoreExports, context) => {
  const srcFiles = resolveFiles(getSrc(src), ignoreExports);
  prepareImportsAndExports(srcFiles, context);
  determineUsage();
  preparationDone = true;
};

const newNamespaceImportExists = specifiers => specifiers.some((_ref4) => {
  let type = _ref4.type;
  return type === IMPORT_NAMESPACE_SPECIFIER;
});

const newDefaultImportExists = specifiers => specifiers.some((_ref5) => {
  let type = _ref5.type;
  return type === IMPORT_DEFAULT_SPECIFIER;
});

module.exports = {
  meta: {
    docs: { url: (0, _docsUrl2.default)('no-unused-modules') },
    schema: [{
      properties: {
        src: {
          description: 'files/paths to be analyzed (only for unused exports)',
          type: 'array',
          minItems: 1,
          items: {
            type: 'string',
            minLength: 1
          }
        },
        ignoreExports: {
          description: 'files/paths for which unused exports will not be reported (e.g module entry points)',
          type: 'array',
          minItems: 1,
          items: {
            type: 'string',
            minLength: 1
          }
        },
        missingExports: {
          description: 'report modules without any exports',
          type: 'boolean'
        },
        unusedExports: {
          description: 'report exports without any usage',
          type: 'boolean'
        }
      },
      not: {
        properties: {
          unusedExports: { enum: [false] },
          missingExports: { enum: [false] }
        }
      },
      anyOf: [{
        not: {
          properties: {
            unusedExports: { enum: [true] }
          }
        },
        required: ['missingExports']
      }, {
        not: {
          properties: {
            missingExports: { enum: [true] }
          }
        },
        required: ['unusedExports']
      }, {
        properties: {
          unusedExports: { enum: [true] }
        },
        required: ['unusedExports']
      }, {
        properties: {
          missingExports: { enum: [true] }
        },
        required: ['missingExports']
      }]
    }]
  },

  create: context => {
    var _ref6 = context.options[0] || {};

    const src = _ref6.src;
    var _ref6$ignoreExports = _ref6.ignoreExports;
    const ignoreExports = _ref6$ignoreExports === undefined ? [] : _ref6$ignoreExports,
          missingExports = _ref6.missingExports,
          unusedExports = _ref6.unusedExports;


    if (unusedExports && !preparationDone) {
      doPreparation(src, ignoreExports, context);
    }

    const file = context.getFilename();

    const checkExportPresence = node => {
      if (!missingExports) {
        return;
      }

      if (ignoredFiles.has(file)) {
        return;
      }

      const exportCount = exportList.get(file);
      const exportAll = exportCount.get(EXPORT_ALL_DECLARATION);
      const namespaceImports = exportCount.get(IMPORT_NAMESPACE_SPECIFIER);

      exportCount.delete(EXPORT_ALL_DECLARATION);
      exportCount.delete(IMPORT_NAMESPACE_SPECIFIER);
      if (missingExports && exportCount.size < 1) {
        // node.body[0] === 'undefined' only happens, if everything is commented out in the file
        // being linted
        context.report(node.body[0] ? node.body[0] : node, 'No exports found');
      }
      exportCount.set(EXPORT_ALL_DECLARATION, exportAll);
      exportCount.set(IMPORT_NAMESPACE_SPECIFIER, namespaceImports);
    };

    const checkUsage = (node, exportedValue) => {
      if (!unusedExports) {
        return;
      }

      if (ignoredFiles.has(file)) {
        return;
      }

      exports = exportList.get(file);

      // special case: export * from 
      const exportAll = exports.get(EXPORT_ALL_DECLARATION);
      if (typeof exportAll !== 'undefined' && exportedValue !== IMPORT_DEFAULT_SPECIFIER) {
        if (exportAll.whereUsed.size > 0) {
          return;
        }
      }

      // special case: namespace import
      const namespaceImports = exports.get(IMPORT_NAMESPACE_SPECIFIER);
      if (typeof namespaceImports !== 'undefined') {
        if (namespaceImports.whereUsed.size > 0) {
          return;
        }
      }

      const exportStatement = exports.get(exportedValue);

      const value = exportedValue === IMPORT_DEFAULT_SPECIFIER ? DEFAULT : exportedValue;

      if (typeof exportStatement !== 'undefined') {
        if (exportStatement.whereUsed.size < 1) {
          context.report(node, `exported declaration '${value}' not used within other modules`);
        }
      } else {
        context.report(node, `exported declaration '${value}' not used within other modules`);
      }
    };

    /**
     * only useful for tools like vscode-eslint
     * 
     * update lists of existing exports during runtime
     */
    const updateExportUsage = node => {
      if (ignoredFiles.has(file)) {
        return;
      }

      let exports = exportList.get(file);

      // new module has been created during runtime
      // include it in further processing
      if (typeof exports === 'undefined') {
        exports = new Map();
      }

      const newExports = new Map();
      const newExportIdentifiers = new Set();

      node.body.forEach((_ref7) => {
        let type = _ref7.type,
            declaration = _ref7.declaration,
            specifiers = _ref7.specifiers;

        if (type === EXPORT_DEFAULT_DECLARATION) {
          newExportIdentifiers.add(IMPORT_DEFAULT_SPECIFIER);
        }
        if (type === EXPORT_NAMED_DECLARATION) {
          if (specifiers.length > 0) {
            specifiers.forEach(specifier => {
              if (specifier.exported) {
                newExportIdentifiers.add(specifier.exported.name);
              }
            });
          }
          if (declaration) {
            if (declaration.type === FUNCTION_DECLARATION) {
              newExportIdentifiers.add(declaration.id.name);
            }
            if (declaration.type === VARIABLE_DECLARATION) {
              declaration.declarations.forEach((_ref8) => {
                let id = _ref8.id;

                newExportIdentifiers.add(id.name);
              });
            }
          }
        }
      });

      // old exports exist within list of new exports identifiers: add to map of new exports
      exports.forEach((value, key) => {
        if (newExportIdentifiers.has(key)) {
          newExports.set(key, value);
        }
      });

      // new export identifiers added: add to map of new exports
      newExportIdentifiers.forEach(key => {
        if (!exports.has(key)) {
          newExports.set(key, { whereUsed: new Set() });
        }
      });

      // preserve information about namespace imports
      let exportAll = exports.get(EXPORT_ALL_DECLARATION);
      let namespaceImports = exports.get(IMPORT_NAMESPACE_SPECIFIER);

      if (typeof namespaceImports === 'undefined') {
        namespaceImports = { whereUsed: new Set() };
      }

      newExports.set(EXPORT_ALL_DECLARATION, exportAll);
      newExports.set(IMPORT_NAMESPACE_SPECIFIER, namespaceImports);
      exportList.set(file, newExports);
    };

    /**
     * only useful for tools like vscode-eslint
     * 
     * update lists of existing imports during runtime
     */
    const updateImportUsage = node => {
      if (!unusedExports) {
        return;
      }

      let oldImportPaths = importList.get(file);
      if (typeof oldImportPaths === 'undefined') {
        oldImportPaths = new Map();
      }

      const oldNamespaceImports = new Set();
      const newNamespaceImports = new Set();

      const oldExportAll = new Set();
      const newExportAll = new Set();

      const oldDefaultImports = new Set();
      const newDefaultImports = new Set();

      const oldImports = new Map();
      const newImports = new Map();
      oldImportPaths.forEach((value, key) => {
        if (value.has(EXPORT_ALL_DECLARATION)) {
          oldExportAll.add(key);
        }
        if (value.has(IMPORT_NAMESPACE_SPECIFIER)) {
          oldNamespaceImports.add(key);
        }
        if (value.has(IMPORT_DEFAULT_SPECIFIER)) {
          oldDefaultImports.add(key);
        }
        value.forEach(val => {
          if (val !== IMPORT_NAMESPACE_SPECIFIER && val !== IMPORT_DEFAULT_SPECIFIER) {
            oldImports.set(val, key);
          }
        });
      });

      node.body.forEach(astNode => {
        let resolvedPath;

        // support for export { value } from 'module'
        if (astNode.type === EXPORT_NAMED_DECLARATION) {
          if (astNode.source) {
            resolvedPath = (0, _resolve2.default)(astNode.source.value, context);
            astNode.specifiers.forEach(specifier => {
              let name;
              if (specifier.exported.name === DEFAULT) {
                name = IMPORT_DEFAULT_SPECIFIER;
              } else {
                name = specifier.local.name;
              }
              newImports.set(name, resolvedPath);
            });
          }
        }

        if (astNode.type === EXPORT_ALL_DECLARATION) {
          resolvedPath = (0, _resolve2.default)(astNode.source.value, context);
          newExportAll.add(resolvedPath);
        }

        if (astNode.type === IMPORT_DECLARATION) {
          resolvedPath = (0, _resolve2.default)(astNode.source.value, context);
          if (!resolvedPath) {
            return;
          }

          if (isNodeModule(resolvedPath)) {
            return;
          }

          if (newNamespaceImportExists(astNode.specifiers)) {
            newNamespaceImports.add(resolvedPath);
          }

          if (newDefaultImportExists(astNode.specifiers)) {
            newDefaultImports.add(resolvedPath);
          }

          astNode.specifiers.forEach(specifier => {
            if (specifier.type === IMPORT_DEFAULT_SPECIFIER || specifier.type === IMPORT_NAMESPACE_SPECIFIER) {
              return;
            }
            newImports.set(specifier.local.name, resolvedPath);
          });
        }
      });

      newExportAll.forEach(value => {
        if (!oldExportAll.has(value)) {
          let imports = oldImportPaths.get(value);
          if (typeof imports === 'undefined') {
            imports = new Set();
          }
          imports.add(EXPORT_ALL_DECLARATION);
          oldImportPaths.set(value, imports);

          let exports = exportList.get(value);
          let currentExport;
          if (typeof exports !== 'undefined') {
            currentExport = exports.get(EXPORT_ALL_DECLARATION);
          } else {
            exports = new Map();
            exportList.set(value, exports);
          }

          if (typeof currentExport !== 'undefined') {
            currentExport.whereUsed.add(file);
          } else {
            const whereUsed = new Set();
            whereUsed.add(file);
            exports.set(EXPORT_ALL_DECLARATION, { whereUsed });
          }
        }
      });

      oldExportAll.forEach(value => {
        if (!newExportAll.has(value)) {
          const imports = oldImportPaths.get(value);
          imports.delete(EXPORT_ALL_DECLARATION);

          const exports = exportList.get(value);
          if (typeof exports !== 'undefined') {
            const currentExport = exports.get(EXPORT_ALL_DECLARATION);
            if (typeof currentExport !== 'undefined') {
              currentExport.whereUsed.delete(file);
            }
          }
        }
      });

      newDefaultImports.forEach(value => {
        if (!oldDefaultImports.has(value)) {
          let imports = oldImportPaths.get(value);
          if (typeof imports === 'undefined') {
            imports = new Set();
          }
          imports.add(IMPORT_DEFAULT_SPECIFIER);
          oldImportPaths.set(value, imports);

          let exports = exportList.get(value);
          let currentExport;
          if (typeof exports !== 'undefined') {
            currentExport = exports.get(IMPORT_DEFAULT_SPECIFIER);
          } else {
            exports = new Map();
            exportList.set(value, exports);
          }

          if (typeof currentExport !== 'undefined') {
            currentExport.whereUsed.add(file);
          } else {
            const whereUsed = new Set();
            whereUsed.add(file);
            exports.set(IMPORT_DEFAULT_SPECIFIER, { whereUsed });
          }
        }
      });

      oldDefaultImports.forEach(value => {
        if (!newDefaultImports.has(value)) {
          const imports = oldImportPaths.get(value);
          imports.delete(IMPORT_DEFAULT_SPECIFIER);

          const exports = exportList.get(value);
          if (typeof exports !== 'undefined') {
            const currentExport = exports.get(IMPORT_DEFAULT_SPECIFIER);
            if (typeof currentExport !== 'undefined') {
              currentExport.whereUsed.delete(file);
            }
          }
        }
      });

      newNamespaceImports.forEach(value => {
        if (!oldNamespaceImports.has(value)) {
          let imports = oldImportPaths.get(value);
          if (typeof imports === 'undefined') {
            imports = new Set();
          }
          imports.add(IMPORT_NAMESPACE_SPECIFIER);
          oldImportPaths.set(value, imports);

          let exports = exportList.get(value);
          let currentExport;
          if (typeof exports !== 'undefined') {
            currentExport = exports.get(IMPORT_NAMESPACE_SPECIFIER);
          } else {
            exports = new Map();
            exportList.set(value, exports);
          }

          if (typeof currentExport !== 'undefined') {
            currentExport.whereUsed.add(file);
          } else {
            const whereUsed = new Set();
            whereUsed.add(file);
            exports.set(IMPORT_NAMESPACE_SPECIFIER, { whereUsed });
          }
        }
      });

      oldNamespaceImports.forEach(value => {
        if (!newNamespaceImports.has(value)) {
          const imports = oldImportPaths.get(value);
          imports.delete(IMPORT_NAMESPACE_SPECIFIER);

          const exports = exportList.get(value);
          if (typeof exports !== 'undefined') {
            const currentExport = exports.get(IMPORT_NAMESPACE_SPECIFIER);
            if (typeof currentExport !== 'undefined') {
              currentExport.whereUsed.delete(file);
            }
          }
        }
      });

      newImports.forEach((value, key) => {
        if (!oldImports.has(key)) {
          let imports = oldImportPaths.get(value);
          if (typeof imports === 'undefined') {
            imports = new Set();
          }
          imports.add(key);
          oldImportPaths.set(value, imports);

          let exports = exportList.get(value);
          let currentExport;
          if (typeof exports !== 'undefined') {
            currentExport = exports.get(key);
          } else {
            exports = new Map();
            exportList.set(value, exports);
          }

          if (typeof currentExport !== 'undefined') {
            currentExport.whereUsed.add(file);
          } else {
            const whereUsed = new Set();
            whereUsed.add(file);
            exports.set(key, { whereUsed });
          }
        }
      });

      oldImports.forEach((value, key) => {
        if (!newImports.has(key)) {
          const imports = oldImportPaths.get(value);
          imports.delete(key);

          const exports = exportList.get(value);
          if (typeof exports !== 'undefined') {
            const currentExport = exports.get(key);
            if (typeof currentExport !== 'undefined') {
              currentExport.whereUsed.delete(file);
            }
          }
        }
      });
    };

    return {
      'Program:exit': node => {
        updateExportUsage(node);
        updateImportUsage(node);
        checkExportPresence(node);
      },
      'ExportDefaultDeclaration': node => {
        checkUsage(node, IMPORT_DEFAULT_SPECIFIER);
      },
      'ExportNamedDeclaration': node => {
        node.specifiers.forEach(specifier => {
          checkUsage(node, specifier.exported.name);
        });
        if (node.declaration) {
          if (node.declaration.type === FUNCTION_DECLARATION) {
            checkUsage(node, node.declaration.id.name);
          }
          if (node.declaration.type === VARIABLE_DECLARATION) {
            node.declaration.declarations.forEach(declaration => {
              checkUsage(node, declaration.id.name);
            });
          }
        }
      }
    };
  }
};
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["rules/no-unused-modules.js"],"names":["listFilesToProcess","require","err","EXPORT_DEFAULT_DECLARATION","EXPORT_NAMED_DECLARATION","EXPORT_ALL_DECLARATION","IMPORT_DECLARATION","IMPORT_NAMESPACE_SPECIFIER","IMPORT_DEFAULT_SPECIFIER","VARIABLE_DECLARATION","FUNCTION_DECLARATION","DEFAULT","preparationDone","importList","Map","exportList","ignoredFiles","Set","isNodeModule","path","test","resolveFiles","src","ignoreExports","srcFiles","srcFileList","ignoredFilesList","forEach","filename","add","filter","prepareImportsAndExports","context","exportAll","file","exports","imports","currentExports","Exports","get","dependencies","reexports","localImportList","namespace","currentExportAll","value","set","key","whereUsed","reexport","getImport","localImport","currentValue","local","importedSpecifiers","has","val","currentExport","determineUsage","listValue","listKey","currentImport","specifier","exportStatement","getSrc","process","cwd","doPreparation","newNamespaceImportExists","specifiers","some","type","newDefaultImportExists","module","meta","docs","url","schema","properties","description","minItems","items","minLength","missingExports","unusedExports","not","enum","anyOf","required","create","options","getFilename","checkExportPresence","node","exportCount","namespaceImports","delete","size","report","body","checkUsage","exportedValue","updateExportUsage","newExports","newExportIdentifiers","declaration","length","exported","name","id","declarations","updateImportUsage","oldImportPaths","oldNamespaceImports","newNamespaceImports","oldExportAll","newExportAll","oldDefaultImports","newDefaultImports","oldImports","newImports","astNode","resolvedPath","source"],"mappings":"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","file":"rules/no-unused-modules.js","sourcesContent":["/**\n * @fileOverview Ensures that modules contain exports and/or all \n * modules are consumed within other modules.\n * @author René Fermann\n */\n\nimport Exports from '../ExportMap'\nimport resolve from 'eslint-module-utils/resolve'\nimport docsUrl from '../docsUrl'\n\n// eslint/lib/util/glob-util has been moved to eslint/lib/util/glob-utils with version 5.3\nlet listFilesToProcess\ntry {\n  listFilesToProcess = require('eslint/lib/util/glob-utils').listFilesToProcess\n} catch (err) {\n  listFilesToProcess = require('eslint/lib/util/glob-util').listFilesToProcess\n}\n\nconst EXPORT_DEFAULT_DECLARATION = 'ExportDefaultDeclaration'\nconst EXPORT_NAMED_DECLARATION = 'ExportNamedDeclaration'\nconst EXPORT_ALL_DECLARATION = 'ExportAllDeclaration'\nconst IMPORT_DECLARATION = 'ImportDeclaration' \nconst IMPORT_NAMESPACE_SPECIFIER = 'ImportNamespaceSpecifier'\nconst IMPORT_DEFAULT_SPECIFIER = 'ImportDefaultSpecifier' \nconst VARIABLE_DECLARATION = 'VariableDeclaration'\nconst FUNCTION_DECLARATION = 'FunctionDeclaration'\nconst DEFAULT = 'default'\n\nlet preparationDone = false\nconst importList = new Map()\nconst exportList = new Map()\nconst ignoredFiles = new Set()\n\nconst isNodeModule = path => {\n  return /\\/(node_modules)\\//.test(path)\n}\n\n/**\n * read all files matching the patterns in src and ignoreExports\n * \n * return all files matching src pattern, which are not matching the ignoreExports pattern\n */\nconst resolveFiles = (src, ignoreExports) => {\n  const srcFiles = new Set()\n  const srcFileList = listFilesToProcess(src)\n\n  // prepare list of ignored files\n  const ignoredFilesList =  listFilesToProcess(ignoreExports)\n  ignoredFilesList.forEach(({ filename }) => ignoredFiles.add(filename))\n\n  // prepare list of source files, don't consider files from node_modules\n  srcFileList.filter(({ filename }) => !isNodeModule(filename)).forEach(({ filename }) => {\n    srcFiles.add(filename)\n  })\n  return srcFiles\n}\n\n/**\n * parse all source files and build up 2 maps containing the existing imports and exports\n */\nconst prepareImportsAndExports = (srcFiles, context) => {\n  const exportAll = new Map()\n  srcFiles.forEach(file => {\n    const exports = new Map()\n    const imports = new Map()\n    const currentExports = Exports.get(file, context)\n    if (currentExports) {\n      const { dependencies, reexports, imports: localImportList, namespace  } = currentExports\n\n      // dependencies === export * from \n      const currentExportAll = new Set()\n      dependencies.forEach(value => {\n        currentExportAll.add(value().path)\n      })\n      exportAll.set(file, currentExportAll)\n      \n      reexports.forEach((value, key) => {\n        if (key === DEFAULT) {\n          exports.set(IMPORT_DEFAULT_SPECIFIER, { whereUsed: new Set() })\n        } else {\n          exports.set(key, { whereUsed: new Set() })\n        }\n        const reexport =  value.getImport()\n        if (!reexport) {\n          return\n        }\n        let localImport = imports.get(reexport.path)\n        let currentValue\n        if (value.local === DEFAULT) {\n          currentValue = IMPORT_DEFAULT_SPECIFIER\n        } else {\n          currentValue = value.local\n        }\n        if (typeof localImport !== 'undefined') {\n          localImport = new Set([...localImport, currentValue])\n        } else {\n          localImport = new Set([currentValue])\n        }\n        imports.set(reexport.path, localImport)\n      })\n\n      localImportList.forEach((value, key) => {\n        if (isNodeModule(key)) {\n          return\n        }\n        imports.set(key, value.importedSpecifiers)\n      })\n      importList.set(file, imports)\n      \n      // build up export list only, if file is not ignored\n      if (ignoredFiles.has(file)) {\n        return\n      }\n      namespace.forEach((value, key) => {\n        if (key === DEFAULT) {\n          exports.set(IMPORT_DEFAULT_SPECIFIER, { whereUsed: new Set() })\n        } else {\n          exports.set(key, { whereUsed: new Set() })\n        }\n      })\n    }\n    exports.set(EXPORT_ALL_DECLARATION, { whereUsed: new Set() })\n    exports.set(IMPORT_NAMESPACE_SPECIFIER, { whereUsed: new Set() })\n    exportList.set(file, exports)\n  })\n  exportAll.forEach((value, key) => {\n    value.forEach(val => {\n      const currentExports = exportList.get(val)\n      const currentExport = currentExports.get(EXPORT_ALL_DECLARATION)\n      currentExport.whereUsed.add(key)\n    })\n  })\n}\n\n/**\n * traverse through all imports and add the respective path to the whereUsed-list \n * of the corresponding export\n */\nconst determineUsage = () => {\n  importList.forEach((listValue, listKey) => {\n    listValue.forEach((value, key) => {\n      const exports = exportList.get(key)\n      if (typeof exports !== 'undefined') {\n        value.forEach(currentImport => {\n          let specifier\n          if (currentImport === IMPORT_NAMESPACE_SPECIFIER) {\n            specifier = IMPORT_NAMESPACE_SPECIFIER\n          } else if (currentImport === IMPORT_DEFAULT_SPECIFIER) {\n            specifier = IMPORT_DEFAULT_SPECIFIER\n          } else {\n            specifier = currentImport\n          }\n          if (typeof specifier !== 'undefined') {\n            const exportStatement = exports.get(specifier)\n            if (typeof exportStatement !== 'undefined') {\n              const { whereUsed } = exportStatement\n              whereUsed.add(listKey)\n              exports.set(specifier, { whereUsed })\n            }\n          }\n        })\n      }\n    })\n  })\n}\n\nconst getSrc = src => {\n  if (src) {\n    return src\n  }\n  return [process.cwd()]\n}\n\n/**\n * prepare the lists of existing imports and exports - should only be executed once at\n * the start of a new eslint run\n */\nconst doPreparation = (src, ignoreExports, context) => {\n  const srcFiles = resolveFiles(getSrc(src), ignoreExports)\n  prepareImportsAndExports(srcFiles, context)\n  determineUsage()\n  preparationDone = true\n}\n\nconst newNamespaceImportExists = specifiers =>\n  specifiers.some(({ type }) => type === IMPORT_NAMESPACE_SPECIFIER)\n\nconst newDefaultImportExists = specifiers =>\n  specifiers.some(({ type }) => type === IMPORT_DEFAULT_SPECIFIER)\n\nmodule.exports = {\n  meta: {\n    docs: { url: docsUrl('no-unused-modules') },\n    schema: [{\n      properties: {\n        src: {\n          description: 'files/paths to be analyzed (only for unused exports)',\n          type: 'array',\n          minItems: 1,\n          items: {\n            type: 'string',\n            minLength: 1,\n          },\n        },\n        ignoreExports: {\n          description:\n            'files/paths for which unused exports will not be reported (e.g module entry points)',\n          type: 'array',\n          minItems: 1,\n          items: {\n            type: 'string',\n            minLength: 1,\n          },\n        },\n        missingExports: {\n          description: 'report modules without any exports',\n          type: 'boolean',\n        },\n        unusedExports: {\n          description: 'report exports without any usage',\n          type: 'boolean',\n        },\n      },\n      not: {\n        properties: {\n          unusedExports: { enum: [false] },\n          missingExports: { enum: [false] },\n        },\n      },\n      anyOf:[{\n        not: {\n          properties: {\n            unusedExports: { enum: [true] },\n          },\n        },\n        required: ['missingExports'],\n      }, {\n        not: {\n          properties: {\n            missingExports: { enum: [true] },\n          },\n        },\n        required: ['unusedExports'],\n      }, {\n        properties: {\n          unusedExports: { enum: [true] },\n        },\n        required: ['unusedExports'],\n      }, {\n        properties: {\n          missingExports: { enum: [true] },\n        },\n        required: ['missingExports'],\n      }],\n    }],\n  },\n\n  create: context => {\n    const {\n      src,\n      ignoreExports = [],\n      missingExports,\n      unusedExports,\n    } = context.options[0] || {}\n\n    if (unusedExports && !preparationDone) {\n      doPreparation(src, ignoreExports, context)\n    }\n    \n    const file = context.getFilename()\n\n    const checkExportPresence = node => {\n      if (!missingExports) {\n        return\n      }\n\n      if (ignoredFiles.has(file)) {\n        return\n      }\n\n      const exportCount = exportList.get(file)\n      const exportAll = exportCount.get(EXPORT_ALL_DECLARATION)\n      const namespaceImports = exportCount.get(IMPORT_NAMESPACE_SPECIFIER)\n\n      exportCount.delete(EXPORT_ALL_DECLARATION)\n      exportCount.delete(IMPORT_NAMESPACE_SPECIFIER)\n      if (missingExports && exportCount.size < 1) {\n        // node.body[0] === 'undefined' only happens, if everything is commented out in the file\n        // being linted\n        context.report(node.body[0] ? node.body[0] : node, 'No exports found')\n      }\n      exportCount.set(EXPORT_ALL_DECLARATION, exportAll)\n      exportCount.set(IMPORT_NAMESPACE_SPECIFIER, namespaceImports)\n    }\n\n    const checkUsage = (node, exportedValue) => {\n      if (!unusedExports) {\n        return\n      }\n\n      if (ignoredFiles.has(file)) {\n        return\n      }\n\n      exports = exportList.get(file)\n\n      // special case: export * from \n      const exportAll = exports.get(EXPORT_ALL_DECLARATION)\n      if (typeof exportAll !== 'undefined' && exportedValue !== IMPORT_DEFAULT_SPECIFIER) {\n        if (exportAll.whereUsed.size > 0) {\n          return\n        }\n      }\n\n      // special case: namespace import\n      const namespaceImports = exports.get(IMPORT_NAMESPACE_SPECIFIER)\n      if (typeof namespaceImports !== 'undefined') {\n        if (namespaceImports.whereUsed.size > 0) {\n          return\n        }\n      }\n\n      const exportStatement = exports.get(exportedValue)\n      \n      const value = exportedValue === IMPORT_DEFAULT_SPECIFIER ? DEFAULT : exportedValue\n      \n      if (typeof exportStatement !== 'undefined'){\n        if (exportStatement.whereUsed.size < 1) {\n          context.report(\n            node,\n            `exported declaration '${value}' not used within other modules`\n          )\n        }\n      } else {\n        context.report(\n          node,\n          `exported declaration '${value}' not used within other modules`\n        )\n      }\n    }\n\n    /**\n     * only useful for tools like vscode-eslint\n     * \n     * update lists of existing exports during runtime\n     */\n    const updateExportUsage = node => {\n      if (ignoredFiles.has(file)) {\n        return\n      }\n\n      let exports = exportList.get(file)\n\n      // new module has been created during runtime\n      // include it in further processing\n      if (typeof exports === 'undefined') {\n        exports = new Map()\n      }\n\n      const newExports = new Map()\n      const newExportIdentifiers = new Set()\n\n      node.body.forEach(({ type, declaration, specifiers }) => {\n        if (type === EXPORT_DEFAULT_DECLARATION) {\n          newExportIdentifiers.add(IMPORT_DEFAULT_SPECIFIER)\n        } \n        if (type === EXPORT_NAMED_DECLARATION) {\n          if (specifiers.length > 0) {\n            specifiers.forEach(specifier => {\n              if (specifier.exported) {\n                newExportIdentifiers.add(specifier.exported.name)\n              }\n            })\n          }\n          if (declaration) {\n            if (declaration.type === FUNCTION_DECLARATION) {\n              newExportIdentifiers.add(declaration.id.name)\n            }   \n            if (declaration.type === VARIABLE_DECLARATION) {\n              declaration.declarations.forEach(({ id }) => {\n                newExportIdentifiers.add(id.name)\n              })\n            }\n          }\n        }\n      })\n\n      // old exports exist within list of new exports identifiers: add to map of new exports\n      exports.forEach((value, key) => {\n        if (newExportIdentifiers.has(key)) {\n          newExports.set(key, value)\n        }\n      })\n\n      // new export identifiers added: add to map of new exports\n      newExportIdentifiers.forEach(key => {\n        if (!exports.has(key)) {\n          newExports.set(key, { whereUsed: new Set() })\n        }\n      })\n\n      // preserve information about namespace imports\n      let exportAll = exports.get(EXPORT_ALL_DECLARATION)\n      let namespaceImports = exports.get(IMPORT_NAMESPACE_SPECIFIER)\n      \n      if (typeof namespaceImports === 'undefined') {\n        namespaceImports = { whereUsed: new Set() }\n      }\n\n      newExports.set(EXPORT_ALL_DECLARATION, exportAll)\n      newExports.set(IMPORT_NAMESPACE_SPECIFIER, namespaceImports)\n      exportList.set(file, newExports)\n    }\n\n    /**\n     * only useful for tools like vscode-eslint\n     * \n     * update lists of existing imports during runtime\n     */\n    const updateImportUsage = node => {\n      if (!unusedExports) {\n        return\n      }\n\n      let oldImportPaths = importList.get(file)\n      if (typeof oldImportPaths === 'undefined') {\n        oldImportPaths = new Map()\n      }\n      \n      const oldNamespaceImports = new Set()\n      const newNamespaceImports = new Set()\n\n      const oldExportAll = new Set()\n      const newExportAll = new Set()\n      \n      const oldDefaultImports = new Set()\n      const newDefaultImports = new Set()\n\n      const oldImports = new Map()\n      const newImports = new Map()\n      oldImportPaths.forEach((value, key) => {\n        if (value.has(EXPORT_ALL_DECLARATION)) {\n          oldExportAll.add(key)\n        }\n        if (value.has(IMPORT_NAMESPACE_SPECIFIER)) {\n          oldNamespaceImports.add(key)\n        }\n        if (value.has(IMPORT_DEFAULT_SPECIFIER)) {\n          oldDefaultImports.add(key)\n        }\n        value.forEach(val => {\n          if (val !== IMPORT_NAMESPACE_SPECIFIER &&\n              val !== IMPORT_DEFAULT_SPECIFIER) {\n               oldImports.set(val, key)\n             }\n        })\n      })\n\n      node.body.forEach(astNode => {\n        let resolvedPath\n\n        // support for export { value } from 'module'\n        if (astNode.type === EXPORT_NAMED_DECLARATION) {\n          if (astNode.source) {\n            resolvedPath = resolve(astNode.source.value, context)\n            astNode.specifiers.forEach(specifier => {\n              let name\n              if (specifier.exported.name === DEFAULT) {\n                name = IMPORT_DEFAULT_SPECIFIER\n              } else {\n                name = specifier.local.name\n              }\n              newImports.set(name, resolvedPath)\n            })\n          }\n        }\n\n        if (astNode.type === EXPORT_ALL_DECLARATION) {\n          resolvedPath = resolve(astNode.source.value, context)\n          newExportAll.add(resolvedPath)\n        }\n\n        if (astNode.type === IMPORT_DECLARATION) {\n          resolvedPath = resolve(astNode.source.value, context)       \n          if (!resolvedPath) {\n            return\n          }\n          \n          if (isNodeModule(resolvedPath)) {\n            return\n          }\n\n          if (newNamespaceImportExists(astNode.specifiers)) {\n            newNamespaceImports.add(resolvedPath)\n          }\n\n          if (newDefaultImportExists(astNode.specifiers)) {\n            newDefaultImports.add(resolvedPath)\n          }\n\n          astNode.specifiers.forEach(specifier => {\n            if (specifier.type === IMPORT_DEFAULT_SPECIFIER ||\n                specifier.type === IMPORT_NAMESPACE_SPECIFIER) {\n              return\n            }\n            newImports.set(specifier.local.name, resolvedPath)\n          })\n        }\n      })\n\n      newExportAll.forEach(value => {\n        if (!oldExportAll.has(value)) {\n          let imports = oldImportPaths.get(value)\n          if (typeof imports === 'undefined') {\n            imports = new Set()\n          }\n          imports.add(EXPORT_ALL_DECLARATION)\n          oldImportPaths.set(value, imports)\n\n          let exports = exportList.get(value)\n          let currentExport\n          if (typeof exports !== 'undefined') {\n            currentExport = exports.get(EXPORT_ALL_DECLARATION)\n          } else {\n            exports = new Map()\n            exportList.set(value, exports)\n          }\n\n          if (typeof currentExport !== 'undefined') {\n            currentExport.whereUsed.add(file)\n          } else {\n            const whereUsed = new Set()\n            whereUsed.add(file)\n            exports.set(EXPORT_ALL_DECLARATION, { whereUsed })\n          }\n        }\n      })\n\n      oldExportAll.forEach(value => {\n        if (!newExportAll.has(value)) {\n          const imports = oldImportPaths.get(value)\n          imports.delete(EXPORT_ALL_DECLARATION)\n\n          const exports = exportList.get(value)\n          if (typeof exports !== 'undefined') {\n            const currentExport = exports.get(EXPORT_ALL_DECLARATION)\n            if (typeof currentExport !== 'undefined') {\n              currentExport.whereUsed.delete(file)\n            }\n          }\n        }\n      })\n\n      newDefaultImports.forEach(value => {\n        if (!oldDefaultImports.has(value)) {\n          let imports = oldImportPaths.get(value)\n          if (typeof imports === 'undefined') {\n            imports = new Set()\n          }\n          imports.add(IMPORT_DEFAULT_SPECIFIER)\n          oldImportPaths.set(value, imports)\n\n          let exports = exportList.get(value)\n          let currentExport\n          if (typeof exports !== 'undefined') {\n            currentExport = exports.get(IMPORT_DEFAULT_SPECIFIER)\n          } else {\n            exports = new Map()\n            exportList.set(value, exports)\n          }\n\n          if (typeof currentExport !== 'undefined') {\n            currentExport.whereUsed.add(file)\n          } else {\n            const whereUsed = new Set()\n            whereUsed.add(file)\n            exports.set(IMPORT_DEFAULT_SPECIFIER, { whereUsed })\n          }\n        }\n      })\n\n      oldDefaultImports.forEach(value => {\n        if (!newDefaultImports.has(value)) {\n          const imports = oldImportPaths.get(value)\n          imports.delete(IMPORT_DEFAULT_SPECIFIER)\n\n          const exports = exportList.get(value)\n          if (typeof exports !== 'undefined') {\n            const currentExport = exports.get(IMPORT_DEFAULT_SPECIFIER)\n            if (typeof currentExport !== 'undefined') {\n              currentExport.whereUsed.delete(file)\n            }\n          }\n        }\n      })\n\n      newNamespaceImports.forEach(value => {\n        if (!oldNamespaceImports.has(value)) {\n          let imports = oldImportPaths.get(value)\n          if (typeof imports === 'undefined') {\n            imports = new Set()\n          }\n          imports.add(IMPORT_NAMESPACE_SPECIFIER)\n          oldImportPaths.set(value, imports)\n\n          let exports = exportList.get(value)\n          let currentExport\n          if (typeof exports !== 'undefined') {\n            currentExport = exports.get(IMPORT_NAMESPACE_SPECIFIER)\n          } else {\n            exports = new Map()\n            exportList.set(value, exports)\n          }\n\n          if (typeof currentExport !== 'undefined') {\n            currentExport.whereUsed.add(file)\n          } else {\n            const whereUsed = new Set()\n            whereUsed.add(file)\n            exports.set(IMPORT_NAMESPACE_SPECIFIER, { whereUsed })\n          }\n        }\n      })\n\n      oldNamespaceImports.forEach(value => {\n        if (!newNamespaceImports.has(value)) {\n          const imports = oldImportPaths.get(value)\n          imports.delete(IMPORT_NAMESPACE_SPECIFIER)\n\n          const exports = exportList.get(value)\n          if (typeof exports !== 'undefined') {\n            const currentExport = exports.get(IMPORT_NAMESPACE_SPECIFIER)\n            if (typeof currentExport !== 'undefined') {\n              currentExport.whereUsed.delete(file)\n            }\n          }\n        }\n      })\n\n      newImports.forEach((value, key) => {\n        if (!oldImports.has(key)) {\n          let imports = oldImportPaths.get(value)\n          if (typeof imports === 'undefined') {\n            imports = new Set()\n          }\n          imports.add(key)\n          oldImportPaths.set(value, imports)\n\n          let exports = exportList.get(value)\n          let currentExport\n          if (typeof exports !== 'undefined') {\n            currentExport = exports.get(key)\n          } else {\n            exports = new Map()\n            exportList.set(value, exports)\n          }\n\n          if (typeof currentExport !== 'undefined') {\n            currentExport.whereUsed.add(file)\n          } else {\n            const whereUsed = new Set()\n            whereUsed.add(file)\n            exports.set(key, { whereUsed })\n          }\n        }\n      })\n\n      oldImports.forEach((value, key) => {\n        if (!newImports.has(key)) {\n          const imports = oldImportPaths.get(value)\n          imports.delete(key)\n\n          const exports = exportList.get(value)\n          if (typeof exports !== 'undefined') {\n            const currentExport = exports.get(key)\n            if (typeof currentExport !== 'undefined') {\n              currentExport.whereUsed.delete(file)\n            }\n          }\n        }\n      })\n    }\n\n    return {\n      'Program:exit': node => {\n        updateExportUsage(node)\n        updateImportUsage(node)\n        checkExportPresence(node)\n      },\n      'ExportDefaultDeclaration': node => {\n        checkUsage(node, IMPORT_DEFAULT_SPECIFIER)\n      },\n      'ExportNamedDeclaration': node => {\n        node.specifiers.forEach(specifier => {\n            checkUsage(node, specifier.exported.name)\n        })\n        if (node.declaration) {\n          if (node.declaration.type === FUNCTION_DECLARATION) {\n            checkUsage(node, node.declaration.id.name)\n          }\n          if (node.declaration.type === VARIABLE_DECLARATION) {\n            node.declaration.declarations.forEach(declaration => {\n              checkUsage(node, declaration.id.name)\n            })\n          }\n        }\n      },\n    }\n  },\n}\n"]}