/** * @author Toru Nagashima <https://github.com/mysticatea> * See LICENSE file in root directory for full license. */ "use strict"; const KEYS = require("./visitor-keys.json"); // Types. const NODE_TYPES = Object.freeze(Object.keys(KEYS)); // Freeze the keys. for (const type of NODE_TYPES) { Object.freeze(KEYS[type]); } Object.freeze(KEYS); // List to ignore keys. const KEY_BLACKLIST = new Set([ "parent", "leadingComments", "trailingComments" ]); /** * Check whether a given key should be used or not. * @param {string} key The key to check. * @returns {boolean} `true` if the key should be used. */ function filterKey(key) { return !KEY_BLACKLIST.has(key) && key[0] !== "_"; } //------------------------------------------------------------------------------ // Public interfaces //------------------------------------------------------------------------------ module.exports = Object.freeze({ /** * Visitor keys. * @type {{ [type: string]: string[] | undefined }} */ KEYS, /** * Get visitor keys of a given node. * @param {Object} node The AST node to get keys. * @returns {string[]} Visitor keys of the node. */ getKeys(node) { return Object.keys(node).filter(filterKey); }, // Disable valid-jsdoc rule because it reports syntax error on the type of @returns. // eslint-disable-next-line valid-jsdoc /** * Make the union set with `KEYS` and given keys. * @param {Object} additionalKeys The additional keys. * @returns {{ [type: string]: string[] | undefined }} The union set. */ unionWith(additionalKeys) { const retv = Object.assign({}, KEYS); for (const type of Object.keys(additionalKeys)) { if (retv.hasOwnProperty(type)) { const keys = new Set(additionalKeys[type]); for (const key of retv[type]) { keys.add(key); } retv[type] = Object.freeze(Array.from(keys)); } else { retv[type] = Object.freeze(Array.from(additionalKeys[type])); } } return Object.freeze(retv); } });