"use strict";
|
/**
|
* @license
|
* Copyright 2017 Palantir Technologies, Inc.
|
*
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
* you may not use this file except in compliance with the License.
|
* You may obtain a copy of the License at
|
*
|
* http://www.apache.org/licenses/LICENSE-2.0
|
*
|
* Unless required by applicable law or agreed to in writing, software
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
* See the License for the specific language governing permissions and
|
* limitations under the License.
|
*/
|
Object.defineProperty(exports, "__esModule", { value: true });
|
var tslib_1 = require("tslib");
|
var tsutils_1 = require("tsutils");
|
var ts = require("typescript");
|
var Lint = require("../index");
|
var Rule = /** @class */ (function (_super) {
|
tslib_1.__extends(Rule, _super);
|
function Rule() {
|
return _super !== null && _super.apply(this, arguments) || this;
|
}
|
Rule.prototype.applyWithProgram = function (sourceFile, program) {
|
return this.applyWithFunction(sourceFile, walk, undefined, program.getTypeChecker());
|
};
|
/* tslint:disable:object-literal-sort-keys */
|
Rule.metadata = {
|
ruleName: "return-undefined",
|
description: "Prefer `return;` in void functions and `return undefined;` in value-returning functions.",
|
optionsDescription: "Not configurable.",
|
options: null,
|
optionExamples: [true],
|
type: "style",
|
typescriptOnly: false,
|
requiresTypeInfo: true,
|
};
|
/* tslint:enable:object-literal-sort-keys */
|
Rule.FAILURE_STRING_VALUE_RETURN = "Value-returning function should use `return undefined;`, not just `return;`.";
|
Rule.FAILURE_STRING_VOID_RETURN = "`void` function should use `return;`, not `return undefined;`.";
|
return Rule;
|
}(Lint.Rules.TypedRule));
|
exports.Rule = Rule;
|
function walk(ctx, checker) {
|
return ts.forEachChild(ctx.sourceFile, function cb(node) {
|
if (tsutils_1.isReturnStatement(node)) {
|
check(node);
|
}
|
return ts.forEachChild(node, cb);
|
});
|
function check(node) {
|
var actualReturnKind = returnKindFromReturn(node);
|
if (actualReturnKind === undefined) {
|
return;
|
}
|
var functionReturningFrom = Lint.ancestorWhere(node, isFunctionLike);
|
if (functionReturningFrom === undefined) {
|
// Return outside of function is invalid
|
return;
|
}
|
var returnKindFromType = getReturnKind(functionReturningFrom, checker);
|
if (returnKindFromType !== undefined && returnKindFromType !== actualReturnKind) {
|
ctx.addFailureAtNode(node, returnKindFromType === ReturnKind.Void
|
? Rule.FAILURE_STRING_VOID_RETURN
|
: Rule.FAILURE_STRING_VALUE_RETURN);
|
}
|
}
|
}
|
function returnKindFromReturn(node) {
|
if (node.expression === undefined) {
|
return ReturnKind.Void;
|
}
|
else if (tsutils_1.isIdentifier(node.expression) && node.expression.text === "undefined") {
|
return ReturnKind.Value;
|
}
|
else {
|
return undefined;
|
}
|
}
|
var ReturnKind;
|
(function (ReturnKind) {
|
ReturnKind[ReturnKind["Void"] = 0] = "Void";
|
ReturnKind[ReturnKind["Value"] = 1] = "Value";
|
})(ReturnKind || (ReturnKind = {}));
|
function getReturnKind(node, checker) {
|
switch (node.kind) {
|
case ts.SyntaxKind.Constructor:
|
case ts.SyntaxKind.SetAccessor:
|
return ReturnKind.Void;
|
case ts.SyntaxKind.GetAccessor:
|
return ReturnKind.Value;
|
}
|
// Handle generator functions/methods:
|
if (node.asteriskToken !== undefined) {
|
return ReturnKind.Void;
|
}
|
var contextual = isFunctionExpressionLike(node) && node.type === undefined
|
? tryGetReturnType(checker.getContextualType(node), checker)
|
: undefined;
|
var returnType = contextual !== undefined
|
? contextual
|
: tryGetReturnType(checker.getTypeAtLocation(node), checker);
|
if (returnType === undefined || tsutils_1.isTypeFlagSet(returnType, ts.TypeFlags.Any)) {
|
return undefined;
|
}
|
if ((tsutils_1.hasModifier(node.modifiers, ts.SyntaxKind.AsyncKeyword)
|
? isEffectivelyVoidPromise
|
: isEffectivelyVoid)(returnType)) {
|
return ReturnKind.Void;
|
}
|
return ReturnKind.Value;
|
}
|
/** True for `void`, `undefined`, Promise<void>, or `void | undefined | Promise<void>`. */
|
function isEffectivelyVoidPromise(type) {
|
// Would need access to `checker.getPromisedTypeOfPromise` to do this properly.
|
// Assume that the return type is the global Promise (since this is an async function) and get its type argument.
|
return (
|
// tslint:disable-next-line:no-bitwise
|
tsutils_1.isTypeFlagSet(type, ts.TypeFlags.Void | ts.TypeFlags.Undefined) ||
|
(tsutils_1.isUnionType(type) && type.types.every(isEffectivelyVoidPromise)) ||
|
(tsutils_1.isTypeReference(type) &&
|
type.typeArguments !== undefined &&
|
type.typeArguments.length === 1 &&
|
isEffectivelyVoidPromise(type.typeArguments[0])));
|
}
|
/** True for `void`, `undefined`, or `void | undefined`. */
|
function isEffectivelyVoid(type) {
|
return (
|
// tslint:disable-next-line:no-bitwise
|
tsutils_1.isTypeFlagSet(type, ts.TypeFlags.Void | ts.TypeFlags.Undefined) ||
|
(tsutils_1.isUnionType(type) && type.types.every(isEffectivelyVoid)));
|
}
|
function tryGetReturnType(fnType, checker) {
|
if (fnType === undefined) {
|
return undefined;
|
}
|
var sigs = checker.getSignaturesOfType(fnType, ts.SignatureKind.Call);
|
if (sigs.length !== 1) {
|
return undefined;
|
}
|
return checker.getReturnTypeOfSignature(sigs[0]);
|
}
|
function isFunctionLike(node) {
|
switch (node.kind) {
|
case ts.SyntaxKind.FunctionDeclaration:
|
case ts.SyntaxKind.FunctionExpression:
|
case ts.SyntaxKind.ArrowFunction:
|
case ts.SyntaxKind.MethodDeclaration:
|
case ts.SyntaxKind.Constructor:
|
case ts.SyntaxKind.GetAccessor:
|
case ts.SyntaxKind.SetAccessor:
|
return true;
|
default:
|
return false;
|
}
|
}
|
function isFunctionExpressionLike(node) {
|
return (node.kind === ts.SyntaxKind.FunctionExpression || node.kind === ts.SyntaxKind.ArrowFunction);
|
}
|