ifc-language-server/node_modules/@stylistic/eslint-plugin/dist/rules/wrap-iife.js
Ryan Schultz 8afacf268a Implemented a working Language Server Protocol (LSP) for IFC files with:
- Hover provider showing entity information and type
- Go-to-definition (F12) for entity references
- Basic IFC file validation (ISO-10303-21 header check)
- Entity parsing with regex-based detection
- Proper CommonJS module system (avoiding ES module issues)

This replaces the broken baseline from ifc-developer-tools which had:
- Non-functional ES module configuration
- Circular dependency issues
- Parser crashes
- Non-working PositionVisitor

Built on Microsoft's LSP example template for a clean, maintainable foundation.

Next: Add hierarchical entity dependency tree in hover tooltip."
2025-12-07 10:20:07 -06:00

117 lines
4 KiB
JavaScript

'use strict';
var utils = require('../utils.js');
require('eslint-visitor-keys');
require('espree');
require('estraverse');
function isCalleeOfNewExpression(node) {
const maybeCallee = node.parent?.type === "ChainExpression" ? node.parent : node;
return maybeCallee.parent?.type === "NewExpression" && maybeCallee.parent.callee === maybeCallee;
}
var wrapIife = utils.createRule({
name: "wrap-iife",
package: "js",
meta: {
type: "layout",
docs: {
description: "Require parentheses around immediate `function` invocations"
},
schema: [
{
type: "string",
enum: ["outside", "inside", "any"]
},
{
type: "object",
properties: {
functionPrototypeMethods: {
type: "boolean",
default: false
}
},
additionalProperties: false
}
],
fixable: "code",
messages: {
wrapInvocation: "Wrap an immediate function invocation in parentheses.",
wrapExpression: "Wrap only the function expression in parens.",
moveInvocation: "Move the invocation into the parens that contain the function."
}
},
create(context) {
const style = context.options[0] || "outside";
const includeFunctionPrototypeMethods = context.options[1] && context.options[1].functionPrototypeMethods;
const sourceCode = context.sourceCode;
function isWrappedInAnyParens(node) {
return utils.isParenthesised(sourceCode, node);
}
function isWrappedInGroupingParens(node) {
return utils.isParenthesized(node, sourceCode, 1);
}
function getFunctionNodeFromIIFE(node) {
const callee = utils.skipChainExpression(node.callee);
if (callee.type === "FunctionExpression")
return callee;
if (includeFunctionPrototypeMethods && callee.type === "MemberExpression" && callee.object.type === "FunctionExpression" && (utils.getStaticPropertyName(callee) === "call" || utils.getStaticPropertyName(callee) === "apply")) {
return callee.object;
}
return null;
}
return {
CallExpression(node) {
const innerNode = getFunctionNodeFromIIFE(node);
if (!innerNode)
return;
const isCallExpressionWrapped = isWrappedInAnyParens(node);
const isFunctionExpressionWrapped = isWrappedInAnyParens(innerNode);
if (!isCallExpressionWrapped && !isFunctionExpressionWrapped) {
context.report({
node,
messageId: "wrapInvocation",
fix(fixer) {
const nodeToSurround = style === "inside" ? innerNode : node;
return fixer.replaceText(nodeToSurround, `(${sourceCode.getText(nodeToSurround)})`);
}
});
} else if (style === "inside" && !isFunctionExpressionWrapped) {
context.report({
node,
messageId: "wrapExpression",
fix(fixer) {
if (isWrappedInGroupingParens(node) && !isCalleeOfNewExpression(node)) {
const parenAfter = sourceCode.getTokenAfter(node);
return fixer.replaceTextRange(
[
innerNode.range[1],
parenAfter.range[1]
],
`)${sourceCode.getText().slice(innerNode.range[1], parenAfter.range[0])}`
);
}
return fixer.replaceText(innerNode, `(${sourceCode.getText(innerNode)})`);
}
});
} else if (style === "outside" && !isCallExpressionWrapped) {
context.report({
node,
messageId: "moveInvocation",
fix(fixer) {
const parenAfter = sourceCode.getTokenAfter(innerNode);
return fixer.replaceTextRange(
[
parenAfter.range[0],
node.range[1]
],
`${sourceCode.getText().slice(parenAfter.range[1], node.range[1])})`
);
}
});
}
}
};
}
});
module.exports = wrapIife;