blob: 39ca1a6e6273e8713676572978be52f36c86c5b9 [file] [log] [blame]
// Copyright (c) 2017, the Dart project authors. Please see the AUTHORS file
// for details. All rights reserved. Use of this source code is governed by a
// BSD-style license that can be found in the LICENSE file.
import 'package:analyzer/dart/ast/ast.dart';
import 'package:analyzer/dart/ast/visitor.dart';
import 'package:analyzer/dart/element/type.dart';
import '../analyzer.dart';
import '../util/dart_type_utilities.dart';
const _desc =
r'Avoid returning null from members whose return type is bool, double, int,'
r' or num.';
const _details = r'''
**AVOID** returning null from members whose return type is bool, double, int,
or num.
Functions that return primitive types such as bool, double, int, and num are
generally expected to return non-nullable values. Thus, returning null where a
primitive type was expected can lead to runtime exceptions.
**BAD:**
```
bool getBool() => null;
num getNum() => null;
int getInt() => null;
double getDouble() => null;
```
**GOOD:**
```
bool getBool() => false;
num getNum() => -1;
int getInt() => -1;
double getDouble() => -1.0;
```
''';
bool _isFunctionExpression(AstNode node) => node is FunctionExpression;
bool _isPrimitiveType(DartType type) =>
DartTypeUtilities.isClass(type, 'bool', 'dart.core') ||
DartTypeUtilities.isClass(type, 'num', 'dart.core') ||
DartTypeUtilities.isClass(type, 'int', 'dart.core') ||
DartTypeUtilities.isClass(type, 'double', 'dart.core');
bool _isReturnNull(AstNode node) =>
node is ReturnStatement && DartTypeUtilities.isNullLiteral(node.expression);
class AvoidReturningNull extends LintRule implements NodeLintRule {
AvoidReturningNull()
: super(
name: 'avoid_returning_null',
description: _desc,
details: _details,
group: Group.style);
@override
void registerNodeProcessors(
NodeLintRegistry registry, LinterContext context) {
final visitor = _Visitor(this);
registry.addFunctionExpression(this, visitor);
registry.addMethodDeclaration(this, visitor);
}
}
class _Visitor extends SimpleAstVisitor<void> {
final LintRule rule;
_Visitor(this.rule);
@override
void visitFunctionExpression(FunctionExpression node) {
if (_isPrimitiveType(node.declaredElement.returnType)) {
_visitFunctionBody(node.body);
}
}
@override
void visitMethodDeclaration(MethodDeclaration node) {
if (_isPrimitiveType(node.declaredElement.returnType)) {
_visitFunctionBody(node.body);
}
}
void _visitFunctionBody(FunctionBody node) {
if (node is ExpressionFunctionBody &&
DartTypeUtilities.isNullLiteral(node.expression)) {
rule.reportLint(node);
return;
}
DartTypeUtilities.traverseNodesInDFS(node,
excludeCriteria: _isFunctionExpression)
.where(_isReturnNull)
.forEach(rule.reportLint);
}
}