"use strict"; | |
Object.defineProperty(exports, "__esModule", { value: true }); | |
exports.callExpressionAffectsControlFlow = exports.SignatureEffect = exports.getControlFlowEnd = exports.endsControlFlow = void 0; | |
const ts = require("typescript"); | |
const node_1 = require("../typeguard/node"); | |
const util_1 = require("./util"); | |
function endsControlFlow(statement, checker) { | |
return getControlFlowEnd(statement, checker).end; | |
} | |
exports.endsControlFlow = endsControlFlow; | |
const defaultControlFlowEnd = { statements: [], end: false }; | |
function getControlFlowEnd(statement, checker) { | |
return node_1.isBlockLike(statement) ? handleBlock(statement, checker) : getControlFlowEndWorker(statement, checker); | |
} | |
exports.getControlFlowEnd = getControlFlowEnd; | |
function getControlFlowEndWorker(statement, checker) { | |
switch (statement.kind) { | |
case ts.SyntaxKind.ReturnStatement: | |
case ts.SyntaxKind.ThrowStatement: | |
case ts.SyntaxKind.ContinueStatement: | |
case ts.SyntaxKind.BreakStatement: | |
return { statements: [statement], end: true }; | |
case ts.SyntaxKind.Block: | |
return handleBlock(statement, checker); | |
case ts.SyntaxKind.ForStatement: | |
case ts.SyntaxKind.WhileStatement: | |
return handleForAndWhileStatement(statement, checker); | |
case ts.SyntaxKind.ForOfStatement: | |
case ts.SyntaxKind.ForInStatement: | |
return handleForInOrOfStatement(statement, checker); | |
case ts.SyntaxKind.DoStatement: | |
return matchBreakOrContinue(getControlFlowEndWorker(statement.statement, checker), node_1.isBreakOrContinueStatement); | |
case ts.SyntaxKind.IfStatement: | |
return handleIfStatement(statement, checker); | |
case ts.SyntaxKind.SwitchStatement: | |
return matchBreakOrContinue(handleSwitchStatement(statement, checker), node_1.isBreakStatement); | |
case ts.SyntaxKind.TryStatement: | |
return handleTryStatement(statement, checker); | |
case ts.SyntaxKind.LabeledStatement: | |
return matchLabel(getControlFlowEndWorker(statement.statement, checker), statement.label); | |
case ts.SyntaxKind.WithStatement: | |
return getControlFlowEndWorker(statement.statement, checker); | |
case ts.SyntaxKind.ExpressionStatement: | |
if (checker === undefined) | |
return defaultControlFlowEnd; | |
return handleExpressionStatement(statement, checker); | |
default: | |
return defaultControlFlowEnd; | |
} | |
} | |
function handleBlock(statement, checker) { | |
const result = { statements: [], end: false }; | |
for (const s of statement.statements) { | |
const current = getControlFlowEndWorker(s, checker); | |
result.statements.push(...current.statements); | |
if (current.end) { | |
result.end = true; | |
break; | |
} | |
} | |
return result; | |
} | |
function handleForInOrOfStatement(statement, checker) { | |
const end = matchBreakOrContinue(getControlFlowEndWorker(statement.statement, checker), node_1.isBreakOrContinueStatement); | |
end.end = false; // loop body is guaranteed to be executed | |
return end; | |
} | |
function handleForAndWhileStatement(statement, checker) { | |
const constantCondition = statement.kind === ts.SyntaxKind.WhileStatement | |
? getConstantCondition(statement.expression) | |
: statement.condition === undefined || getConstantCondition(statement.condition); | |
if (constantCondition === false) | |
return defaultControlFlowEnd; // loop body is never executed | |
const end = matchBreakOrContinue(getControlFlowEndWorker(statement.statement, checker), node_1.isBreakOrContinueStatement); | |
if (constantCondition === undefined) | |
end.end = false; // can't be sure that loop body is executed at all | |
return end; | |
} | |
/** Simply detects `true` and `false` in conditions. That matches TypeScript's behavior. */ | |
function getConstantCondition(node) { | |
switch (node.kind) { | |
case ts.SyntaxKind.TrueKeyword: | |
return true; | |
case ts.SyntaxKind.FalseKeyword: | |
return false; | |
default: | |
return; | |
} | |
} | |
function handleIfStatement(node, checker) { | |
switch (getConstantCondition(node.expression)) { | |
case true: | |
// else branch is never executed | |
return getControlFlowEndWorker(node.thenStatement, checker); | |
case false: | |
// then branch is never executed | |
return node.elseStatement === undefined | |
? defaultControlFlowEnd | |
: getControlFlowEndWorker(node.elseStatement, checker); | |
} | |
const then = getControlFlowEndWorker(node.thenStatement, checker); | |
if (node.elseStatement === undefined) | |
return { | |
statements: then.statements, | |
end: false, | |
}; | |
const elze = getControlFlowEndWorker(node.elseStatement, checker); | |
return { | |
statements: [...then.statements, ...elze.statements], | |
end: then.end && elze.end, | |
}; | |
} | |
function handleSwitchStatement(node, checker) { | |
let hasDefault = false; | |
const result = { | |
statements: [], | |
end: false, | |
}; | |
for (const clause of node.caseBlock.clauses) { | |
if (clause.kind === ts.SyntaxKind.DefaultClause) | |
hasDefault = true; | |
const current = handleBlock(clause, checker); | |
result.end = current.end; | |
result.statements.push(...current.statements); | |
} | |
result.end && (result.end = hasDefault || checker !== undefined && util_1.hasExhaustiveCaseClauses(node, checker)); | |
return result; | |
} | |
function handleTryStatement(node, checker) { | |
let finallyResult; | |
if (node.finallyBlock !== undefined) { | |
finallyResult = handleBlock(node.finallyBlock, checker); | |
// if 'finally' always ends control flow, we are not interested in any jump statements from 'try' or 'catch' | |
if (finallyResult.end) | |
return finallyResult; | |
} | |
const tryResult = handleBlock(node.tryBlock, checker); | |
if (node.catchClause === undefined) | |
return { statements: finallyResult.statements.concat(tryResult.statements), end: tryResult.end }; | |
const catchResult = handleBlock(node.catchClause.block, checker); | |
return { | |
statements: tryResult.statements | |
// remove all throw statements and throwing function calls from the list of control flow statements inside tryBlock | |
.filter((s) => s.kind !== ts.SyntaxKind.ThrowStatement && s.kind !== ts.SyntaxKind.ExpressionStatement) | |
.concat(catchResult.statements, finallyResult === undefined ? [] : finallyResult.statements), | |
end: tryResult.end && catchResult.end, // only ends control flow if try AND catch definitely end control flow | |
}; | |
} | |
/** Dotted name as TypeScript requires it for assertion signatures to affect control flow. */ | |
function isDottedNameWithExplicitTypeAnnotation(node, checker) { | |
while (true) { | |
switch (node.kind) { | |
case ts.SyntaxKind.Identifier: { | |
const symbol = checker.getExportSymbolOfSymbol(checker.getSymbolAtLocation(node)); | |
return isExplicitlyTypedSymbol(util_1.isSymbolFlagSet(symbol, ts.SymbolFlags.Alias) ? checker.getAliasedSymbol(symbol) : symbol, checker); | |
} | |
case ts.SyntaxKind.ThisKeyword: | |
return isExplicitlyTypedThis(node); | |
case ts.SyntaxKind.SuperKeyword: | |
return true; | |
case ts.SyntaxKind.PropertyAccessExpression: | |
if (!isExplicitlyTypedSymbol(checker.getSymbolAtLocation(node), checker)) | |
return false; | |
// falls through | |
case ts.SyntaxKind.ParenthesizedExpression: | |
node = node.expression; | |
continue; | |
default: | |
return false; | |
} | |
} | |
} | |
function isExplicitlyTypedSymbol(symbol, checker) { | |
if (symbol === undefined) | |
return false; | |
if (util_1.isSymbolFlagSet(symbol, ts.SymbolFlags.Function | ts.SymbolFlags.Method | ts.SymbolFlags.Class | ts.SymbolFlags.ValueModule)) | |
return true; | |
if (!util_1.isSymbolFlagSet(symbol, ts.SymbolFlags.Variable | ts.SymbolFlags.Property)) | |
return false; | |
if (symbol.valueDeclaration === undefined) | |
return false; | |
if (declarationHasExplicitTypeAnnotation(symbol.valueDeclaration)) | |
return true; | |
return node_1.isVariableDeclaration(symbol.valueDeclaration) && | |
symbol.valueDeclaration.parent.parent.kind === ts.SyntaxKind.ForOfStatement && | |
isDottedNameWithExplicitTypeAnnotation(symbol.valueDeclaration.parent.parent.expression, checker); | |
} | |
function declarationHasExplicitTypeAnnotation(node) { | |
if (ts.isJSDocPropertyLikeTag(node)) | |
return node.typeExpression !== undefined; | |
return (node_1.isVariableDeclaration(node) || | |
node_1.isParameterDeclaration(node) || | |
node_1.isPropertyDeclaration(node) || | |
node_1.isPropertySignature(node)) && (util_1.isNodeFlagSet(node, ts.NodeFlags.JavaScriptFile) | |
? ts.getJSDocType(node) | |
: node.type) !== undefined; | |
} | |
function isExplicitlyTypedThis(node) { | |
var _a; | |
do { | |
node = node.parent; | |
if (node_1.isDecorator(node)) { | |
// `this` in decorators always resolves outside of the containing class | |
if (node.parent.kind === ts.SyntaxKind.Parameter && node_1.isClassLikeDeclaration(node.parent.parent.parent)) { | |
node = node.parent.parent.parent.parent; | |
} | |
else if (node_1.isClassLikeDeclaration(node.parent.parent)) { | |
node = node.parent.parent.parent; | |
} | |
else if (node_1.isClassLikeDeclaration(node.parent)) { | |
node = node.parent.parent; | |
} | |
} | |
} while (util_1.isFunctionScopeBoundary(node) !== 1 /* Function */ || node.kind === ts.SyntaxKind.ArrowFunction); | |
return util_1.isFunctionWithBody(node) && | |
(util_1.isNodeFlagSet(node, ts.NodeFlags.JavaScriptFile) | |
? ((_a = ts.getJSDocThisTag(node)) === null || _a === void 0 ? void 0 : _a.typeExpression) !== undefined | |
: node.parameters.length !== 0 && util_1.isThisParameter(node.parameters[0]) && node.parameters[0].type !== undefined) || | |
node_1.isClassLikeDeclaration(node.parent); | |
} | |
var SignatureEffect; | |
(function (SignatureEffect) { | |
SignatureEffect[SignatureEffect["Never"] = 1] = "Never"; | |
SignatureEffect[SignatureEffect["Asserts"] = 2] = "Asserts"; | |
})(SignatureEffect = exports.SignatureEffect || (exports.SignatureEffect = {})); | |
/** | |
* Dermines whether a top level CallExpression has a control flow effect according to TypeScript's rules. | |
* This handles functions returning `never` and `asserts`. | |
*/ | |
function callExpressionAffectsControlFlow(node, checker) { | |
var _a, _b, _c; | |
if (!node_1.isExpressionStatement(node.parent) || | |
ts.isOptionalChain(node) || | |
!isDottedNameWithExplicitTypeAnnotation(node.expression, checker)) | |
return; | |
const signature = checker.getResolvedSignature(node); | |
if ((signature === null || signature === void 0 ? void 0 : signature.declaration) === undefined) | |
return; | |
const typeNode = ts.isJSDocSignature(signature.declaration) | |
? (_b = (_a = signature.declaration.type) === null || _a === void 0 ? void 0 : _a.typeExpression) === null || _b === void 0 ? void 0 : _b.type | |
: (_c = signature.declaration.type) !== null && _c !== void 0 ? _c : (util_1.isNodeFlagSet(signature.declaration, ts.NodeFlags.JavaScriptFile) | |
? ts.getJSDocReturnType(signature.declaration) | |
: undefined); | |
if (typeNode === undefined) | |
return; | |
if (node_1.isTypePredicateNode(typeNode) && typeNode.assertsModifier !== undefined) | |
return 2 /* Asserts */; | |
return util_1.isTypeFlagSet(checker.getTypeFromTypeNode(typeNode), ts.TypeFlags.Never) ? 1 /* Never */ : undefined; | |
} | |
exports.callExpressionAffectsControlFlow = callExpressionAffectsControlFlow; | |
function handleExpressionStatement(node, checker) { | |
if (!node_1.isCallExpression(node.expression)) | |
return defaultControlFlowEnd; | |
switch (callExpressionAffectsControlFlow(node.expression, checker)) { | |
case 2 /* Asserts */: | |
return { statements: [node], end: false }; | |
case 1 /* Never */: | |
return { statements: [node], end: true }; | |
case undefined: | |
return defaultControlFlowEnd; | |
} | |
} | |
function matchBreakOrContinue(current, pred) { | |
const result = { | |
statements: [], | |
end: current.end, | |
}; | |
for (const statement of current.statements) { | |
if (pred(statement) && statement.label === undefined) { | |
result.end = false; | |
continue; | |
} | |
result.statements.push(statement); | |
} | |
return result; | |
} | |
function matchLabel(current, label) { | |
const result = { | |
statements: [], | |
end: current.end, | |
}; | |
const labelText = label.text; | |
for (const statement of current.statements) { | |
switch (statement.kind) { | |
case ts.SyntaxKind.BreakStatement: | |
case ts.SyntaxKind.ContinueStatement: | |
if (statement.label !== undefined && statement.label.text === labelText) { | |
result.end = false; | |
continue; | |
} | |
} | |
result.statements.push(statement); | |
} | |
return result; | |
} | |
//# sourceMappingURL=control-flow.js.map |