| // Copyright (c) 2019, 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 'dart:math' as math; |
| |
| import 'package:analyzer/dart/ast/ast.dart'; |
| import 'package:analyzer/dart/element/element.dart'; |
| import 'package:analyzer/dart/element/nullability_suffix.dart'; |
| import 'package:analyzer/dart/element/type.dart'; |
| import 'package:analyzer/dart/element/type_provider.dart'; |
| import 'package:analyzer/src/dart/element/element.dart'; |
| import 'package:analyzer/src/dart/element/member.dart'; |
| import 'package:analyzer/src/dart/element/type.dart'; |
| import 'package:analyzer/src/generated/element_type_provider.dart'; |
| import 'package:analyzer/src/generated/source.dart'; |
| import 'package:analyzer/src/generated/utilities_dart.dart'; |
| import 'package:meta/meta.dart'; |
| import 'package:nnbd_migration/instrumentation.dart'; |
| import 'package:nnbd_migration/src/already_migrated_code_decorator.dart'; |
| import 'package:nnbd_migration/src/conditional_discard.dart'; |
| import 'package:nnbd_migration/src/decorated_type.dart'; |
| import 'package:nnbd_migration/src/expression_checks.dart'; |
| import 'package:nnbd_migration/src/fix_builder.dart'; |
| import 'package:nnbd_migration/src/nullability_node.dart'; |
| import 'package:nnbd_migration/src/nullability_node_target.dart'; |
| import 'package:nnbd_migration/src/utilities/hint_utils.dart'; |
| |
| /// Data structure used by [Variables.spanForUniqueIdentifier] to return an |
| /// offset/end pair. |
| class OffsetEndPair { |
| final int offset; |
| |
| final int end; |
| |
| OffsetEndPair(this.offset, this.end); |
| |
| @override |
| String toString() => '$offset-$end'; |
| } |
| |
| class Variables { |
| final NullabilityGraph _graph; |
| |
| final TypeProvider _typeProvider; |
| |
| final _conditionalDiscards = <Source?, Map<int, ConditionalDiscard>>{}; |
| |
| final _decoratedElementTypes = <Element?, DecoratedType?>{}; |
| |
| final _decoratedDirectSupertypes = |
| <ClassElement, Map<ClassElement, DecoratedType?>>{}; |
| |
| final _decoratedTypeAnnotations = <Source?, Map<int, DecoratedType>>{}; |
| |
| final _expressionChecks = <Source?, Map<int, ExpressionChecks>>{}; |
| |
| final _lateHints = <Source?, Map<int, HintComment>>{}; |
| |
| final _nullCheckHints = <Source?, Map<int, HintComment>>{}; |
| |
| final _nullabilityHints = <Source?, Map<int, HintComment>>{}; |
| |
| final _requiredHints = <Source?, Map<int, HintComment>>{}; |
| |
| final _unnecessaryCasts = <Source?, Set<int>>{}; |
| |
| final AlreadyMigratedCodeDecorator _alreadyMigratedCodeDecorator; |
| |
| final NullabilityMigrationInstrumentation? instrumentation; |
| |
| Variables(this._graph, this._typeProvider, {this.instrumentation}) |
| : _alreadyMigratedCodeDecorator = |
| AlreadyMigratedCodeDecorator(_graph, _typeProvider); |
| |
| /// Given a [class_], gets the decorated type information for the superclasses |
| /// it directly implements/extends/etc. |
| Map<ClassElement, DecoratedType?> decoratedDirectSupertypes( |
| ClassElement class_) { |
| return _decoratedDirectSupertypes[class_] ??= |
| _decorateDirectSupertypes(class_); |
| } |
| |
| /// Retrieves the [DecoratedType] associated with the static type of the given |
| /// [element]. |
| /// |
| /// If no decorated type is found for the given element, and the element is in |
| /// a library that's not being migrated, a decorated type is synthesized using |
| /// [DecoratedType.forElement]. |
| DecoratedType decoratedElementType(Element element) { |
| assert(element is! TypeParameterElement, |
| 'Use decoratedTypeParameterBound instead'); |
| return _decoratedElementTypes[element] ??= |
| _createDecoratedElementType(element); |
| } |
| |
| /// Gets the [DecoratedType] associated with the given [typeAnnotation]. |
| DecoratedType decoratedTypeAnnotation( |
| Source? source, TypeAnnotation typeAnnotation) { |
| var annotationsInSource = _decoratedTypeAnnotations[source]; |
| if (annotationsInSource == null) { |
| throw StateError('No declarated type annotations in ${source!.fullName}; ' |
| 'expected one for ${typeAnnotation.toSource()} ' |
| '(offset ${typeAnnotation.offset})'); |
| } |
| DecoratedType? decoratedTypeAnnotation = annotationsInSource[ |
| uniqueIdentifierForSpan(typeAnnotation.offset, typeAnnotation.end)]; |
| if (decoratedTypeAnnotation == null) { |
| throw StateError('Missing declarated type annotation' |
| ' in ${source!.fullName}; for ${typeAnnotation.toSource()}'); |
| } |
| return decoratedTypeAnnotation; |
| } |
| |
| /// Retrieves the decorated bound of the given [typeParameter]. |
| /// |
| /// Note: the optional argument [allowNullUnparentedBounds] is intended for |
| /// the FixBuilder stage only, to allow it to cope with the situation where |
| /// a type parameter element with a null parent doesn't have a decorated type |
| /// associated with it. This can arise because synthetic type parameter |
| /// elements get created as a result of type system operations during |
| /// resolution, and fortunately it isn't a problem during the FixBuilder stage |
| /// because at that point the types we are dealing with are all |
| /// post-migration types, so their bounds already reflect the correct |
| /// nullabilities. |
| DecoratedType? decoratedTypeParameterBound(TypeParameterElement typeParameter, |
| {bool allowNullUnparentedBounds = false}) { |
| var enclosingElement = typeParameter.enclosingElement; |
| var decoratedType = |
| DecoratedTypeParameterBounds.current!.get(typeParameter); |
| if (enclosingElement == null) { |
| if (decoratedType == null && !allowNullUnparentedBounds) { |
| throw StateError( |
| 'A decorated type for the bound of $typeParameter should ' |
| 'have been stored by the NodeBuilder via recordTypeParameterBound'); |
| } |
| } else { |
| if (decoratedType == null) { |
| if (_graph.isBeingMigrated(typeParameter.library!.source)) { |
| throw StateError( |
| 'A decorated type for the bound of $typeParameter should ' |
| 'have been stored by the NodeBuilder via ' |
| 'recordTypeParameterBound'); |
| } |
| var target = NullabilityNodeTarget.typeParameterBound(typeParameter); |
| decoratedType = _alreadyMigratedCodeDecorator.decorate( |
| typeParameter.preMigrationBound ?? DynamicTypeImpl.instance, |
| typeParameter, |
| target); |
| instrumentation?.externalDecoratedTypeParameterBound( |
| typeParameter, decoratedType); |
| DecoratedTypeParameterBounds.current!.put(typeParameter, decoratedType); |
| } |
| } |
| return decoratedType; |
| } |
| |
| /// Retrieves the [ExpressionChecks] object corresponding to the given |
| /// [expression], if one exists; otherwise null. |
| ExpressionChecks? expressionChecks(Source? source, Expression expression) { |
| return (_expressionChecks[source] ?? |
| {})[uniqueIdentifierForSpan(expression.offset, expression.end)]; |
| } |
| |
| ConditionalDiscard? getConditionalDiscard(Source? source, AstNode node) => |
| (_conditionalDiscards[source] ?? {})[node.offset]; |
| |
| /// If the given [node] is preceded by a `/*late*/` hint, returns the |
| /// HintComment for it; otherwise returns `null`. See [recordLateHint]. |
| HintComment? getLateHint(Source? source, VariableDeclarationList node) { |
| return (_lateHints[source] ?? {})[node.offset]; |
| } |
| |
| /// If the given [node] is followed by a `/*?*/` or /*!*/ hint, returns the |
| /// HintComment for it; otherwise returns `null`. See |
| /// [recordNullabilityHint]. |
| HintComment? getNullabilityHint(Source? source, AstNode node) { |
| assert(node is TypeAnnotation || |
| node is FunctionTypedFormalParameter || |
| (node is FieldFormalParameter && node.parameters != null)); |
| return (_nullabilityHints[source] ?? |
| {})[uniqueIdentifierForSpan(node.offset, node.end)]; |
| } |
| |
| /// If the given [expression] is followed by a null check hint (`/*!*/`), |
| /// returns the HintComment for it; otherwise returns `null`. See |
| /// [recordNullCheckHint]. |
| HintComment? getNullCheckHint(Source? source, Expression expression) { |
| return (_nullCheckHints[source] ?? |
| {})[(uniqueIdentifierForSpan(expression.offset, expression.end))]; |
| } |
| |
| /// If the given [node] is preceded by a `/*required*/` hint, returns the |
| /// HintComment for it; otherwise returns `null`. See [recordRequiredHint]. |
| HintComment? getRequiredHint(Source? source, FormalParameter node) { |
| return (_requiredHints[source] ?? {})[node.offset]; |
| } |
| |
| /// Records conditional discard information for the given AST node (which is |
| /// an `if` statement or a conditional (`?:`) expression). |
| void recordConditionalDiscard( |
| Source? source, AstNode node, ConditionalDiscard conditionalDiscard) { |
| (_conditionalDiscards[source] ??= {})[node.offset] = conditionalDiscard; |
| } |
| |
| /// Associates a [class_] with decorated type information for the superclasses |
| /// it directly implements/extends/etc. |
| void recordDecoratedDirectSupertypes(ClassElement class_, |
| Map<ClassElement, DecoratedType?> decoratedDirectSupertypes) { |
| _decoratedDirectSupertypes[class_] = decoratedDirectSupertypes; |
| } |
| |
| /// Associates decorated type information with the given [element]. |
| void recordDecoratedElementType(Element? element, DecoratedType? type, |
| {bool soft = false}) { |
| assert(() { |
| assert(element is! TypeParameterElement, |
| 'Use recordDecoratedTypeParameterBound instead'); |
| var library = element!.library; |
| if (library == null) { |
| // No problem; the element is probably a parameter of a function type |
| // expressed using new-style Function syntax. |
| } else { |
| assert(_graph.isBeingMigrated(library.source)); |
| } |
| return true; |
| }()); |
| if (soft && _decoratedElementTypes.containsKey(element)) { |
| return; |
| } |
| _decoratedElementTypes[element] = type; |
| } |
| |
| /// Associates decorated type information with the given expression [node]. |
| void recordDecoratedExpressionType(Expression node, DecoratedType? type) {} |
| |
| /// Associates decorated type information with the given [type] node. |
| void recordDecoratedTypeAnnotation( |
| Source? source, TypeAnnotation node, DecoratedType type) { |
| instrumentation?.explicitTypeNullability(source, node, type.node); |
| var id = uniqueIdentifierForSpan(node.offset, node.end); |
| (_decoratedTypeAnnotations[source] ??= {})[id] = type; |
| } |
| |
| /// Associates a set of nullability checks with the given expression [node]. |
| void recordExpressionChecks( |
| Source? source, Expression expression, ExpressionChecksOrigin origin) { |
| (_expressionChecks[source] ??= |
| {})[uniqueIdentifierForSpan(expression.offset, expression.end)] = |
| origin.checks; |
| } |
| |
| /// Records that the given [node] was preceded by a `/*late*/` hint. |
| void recordLateHint( |
| Source? source, VariableDeclarationList node, HintComment hint) { |
| (_lateHints[source] ??= {})[node.offset] = hint; |
| } |
| |
| /// Records that the given [node] was followed by a `/*?*/` or `/*!*/` hint. |
| void recordNullabilityHint( |
| Source? source, AstNode node, HintComment hintComment) { |
| assert(node is TypeAnnotation || |
| node is FunctionTypedFormalParameter || |
| (node is FieldFormalParameter && node.parameters != null)); |
| (_nullabilityHints[source] ??= |
| {})[uniqueIdentifierForSpan(node.offset, node.end)] = hintComment; |
| } |
| |
| /// Records that the given [expression] is followed by a null check hint |
| /// (`/*!*/`), for later recall by [hasNullCheckHint]. |
| void recordNullCheckHint( |
| Source? source, Expression expression, HintComment hintComment) { |
| (_nullCheckHints[source] ??= |
| {})[uniqueIdentifierForSpan(expression.offset, expression.end)] = |
| hintComment; |
| } |
| |
| /// Records that the given [node] was preceded by a `/*required*/` hint. |
| void recordRequiredHint( |
| Source? source, FormalParameter node, HintComment hint) { |
| (_requiredHints[source] ??= {})[node.offset] = hint; |
| } |
| |
| /// Records the fact that prior to migration, an unnecessary cast existed at |
| /// [node]. |
| void recordUnnecessaryCast(Source? source, AsExpression node) { |
| bool newlyAdded = (_unnecessaryCasts[source] ??= {}) |
| .add(uniqueIdentifierForSpan(node.offset, node.end)); |
| assert(newlyAdded); |
| } |
| |
| /// Convert this decorated type into the [DartType] that it will represent |
| /// after the code has been migrated. |
| /// |
| /// This method should be used after nullability propagation; it makes use of |
| /// the nullabilities associated with nullability nodes to determine which |
| /// types should be nullable and which types should not. |
| DartType toFinalType(DecoratedType decoratedType) { |
| var type = decoratedType.type!; |
| if (type.isVoid || type.isDynamic) return type; |
| if (type is NeverType) { |
| if (decoratedType.node!.isNullable) { |
| return (_typeProvider.nullType as TypeImpl) |
| .withNullability(NullabilitySuffix.none); |
| } else { |
| return NeverTypeImpl.instance; |
| } |
| } else if (type.isDartCoreNull) { |
| return (_typeProvider.nullType as TypeImpl) |
| .withNullability(NullabilitySuffix.none); |
| } |
| var nullabilitySuffix = decoratedType.node!.isNullable |
| ? NullabilitySuffix.question |
| : NullabilitySuffix.none; |
| if (type is FunctionType) { |
| var parameters = <ParameterElement>[]; |
| for (int i = 0; i < type.parameters.length; i++) { |
| var origParameter = type.parameters[i]; |
| ParameterKind parameterKind; |
| DecoratedType? parameterType; |
| var name = origParameter.name; |
| if (origParameter.isNamed) { |
| // TODO(paulberry): infer ParameterKind.NAMED_REQUIRED when |
| // appropriate. See https://github.com/dart-lang/sdk/issues/38596. |
| parameterKind = ParameterKind.NAMED; |
| parameterType = decoratedType.namedParameters![name]; |
| } else { |
| parameterKind = origParameter.isOptional |
| ? ParameterKind.POSITIONAL |
| : ParameterKind.REQUIRED; |
| parameterType = decoratedType.positionalParameters![i]; |
| } |
| parameters.add(ParameterElementImpl.synthetic( |
| name, toFinalType(parameterType!), parameterKind)); |
| } |
| return FunctionTypeImpl( |
| typeFormals: type.typeFormals, |
| parameters: parameters, |
| returnType: toFinalType(decoratedType.returnType!), |
| nullabilitySuffix: nullabilitySuffix, |
| ); |
| } else if (type is InterfaceType) { |
| return InterfaceTypeImpl( |
| element: type.element, |
| typeArguments: [ |
| for (var arg in decoratedType.typeArguments) toFinalType(arg!) |
| ], |
| nullabilitySuffix: nullabilitySuffix, |
| ); |
| } else if (type is TypeParameterType) { |
| return TypeParameterTypeImpl( |
| element: type.element, |
| nullabilitySuffix: nullabilitySuffix, |
| ); |
| } else { |
| // The above cases should cover all possible types. On the off chance |
| // they don't, fall back on returning DecoratedType.type. |
| assert(false, 'Unexpected type (${type.runtimeType})'); |
| return type; |
| } |
| } |
| |
| /// Queries whether, prior to migration, an unnecessary cast existed at |
| /// [node]. |
| bool wasUnnecessaryCast(Source? source, AsExpression node) => |
| (_unnecessaryCasts[source] ?? const {}) |
| .contains(uniqueIdentifierForSpan(node.offset, node.end)); |
| |
| /// Creates a decorated type for the given [element], which should come from |
| /// an already-migrated library (or the SDK). |
| DecoratedType _createDecoratedElementType(Element element) { |
| if (_graph.isBeingMigrated(element.library!.source) && |
| !_isLoadLibraryElement(element)) { |
| var description; |
| if (ElementTypeProvider.current is MigrationResolutionHooksImpl) { |
| // Don't attempt to call toString() on element, or we will overflow. |
| description = element.location; |
| } else { |
| description = element; |
| } |
| throw StateError( |
| 'A decorated type for $description should have been stored ' |
| 'by the NodeBuilder via recordDecoratedElementType'); |
| } |
| |
| DecoratedType decoratedType; |
| if (element is Member) { |
| assert(element.isLegacy); |
| element = element.declaration; |
| } |
| |
| if (element is TypeAliasElement) { |
| // For `typedef F<T> = Function(T)`, get the `function` which is (in this |
| // case) `Function(T)`. Without this we would get `Function<T>(T)` which |
| // is incorrect. This is a known issue with `.type` on typedefs in the |
| // analyzer. |
| element = element.aliasedElement!; |
| } |
| |
| var target = NullabilityNodeTarget.element(element); |
| if (element is FunctionTypedElement) { |
| decoratedType = _alreadyMigratedCodeDecorator.decorate( |
| element.preMigrationType, element, target); |
| } else if (element is VariableElement) { |
| decoratedType = _alreadyMigratedCodeDecorator.decorate( |
| element.preMigrationType, element, target); |
| } else if (element is ExtensionElement) { |
| decoratedType = _alreadyMigratedCodeDecorator.decorate( |
| element.preMigrationExtendedType, element, target); |
| } else { |
| // TODO(paulberry) |
| throw UnimplementedError('Decorating ${element.runtimeType}'); |
| } |
| instrumentation?.externalDecoratedType(element, decoratedType); |
| return decoratedType; |
| } |
| |
| /// Creates an entry [_decoratedDirectSupertypes] for an already-migrated |
| /// class. |
| Map<ClassElement, DecoratedType> _decorateDirectSupertypes( |
| ClassElement class_) { |
| var result = <ClassElement, DecoratedType>{}; |
| for (var decoratedSupertype |
| in _alreadyMigratedCodeDecorator.getImmediateSupertypes(class_)) { |
| var class_ = (decoratedSupertype.type as InterfaceType).element; |
| result[class_] = decoratedSupertype; |
| } |
| return result; |
| } |
| |
| bool _isLoadLibraryElement(Element element) => |
| element.isSynthetic && |
| element is FunctionElement && |
| element.enclosingElement is LibraryElement && |
| element.name == 'loadLibrary'; |
| |
| /// Inverts the logic of [uniqueIdentifierForSpan], producing an (offset, end) |
| /// pair. |
| static OffsetEndPair spanForUniqueIdentifier(int span) { |
| // The formula for uniqueIdentifierForSpan was: |
| // span = end*(end + 1) / 2 + offset |
| // In other words, all encodings with the same `end` value are consecutive. |
| // So we just have to figure out the `end` value for this `span`, then |
| // use [uniqueIdentifierForSpan] to find the first encoding with this `end` |
| // value, and subtract to find the offset. |
| // |
| // To find the `end` value, we assume offset = 0 and solve for `end` using |
| // the quadratic formula: |
| // span = end*(end + 1) / 2 |
| // end^2 + end - 2*span = 0 |
| // end = -1 +/- sqrt(1 + 8*span) |
| // We can reslove the `+/-` to `+` (since the result we seek can't be |
| // negative), so that yields: |
| // end = sqrt(1 + 8*span) - 1 |
| int end = (math.sqrt(1 + 8.0 * span) - 1).floor(); |
| assert(end >= 0); |
| |
| // There's a slight chance of numerical instabilities in `sqrt` leading to |
| // a result for `end` that's off by 1, so we loop to find the correct |
| // result: |
| while (true) { |
| // Compute the first `span` value corresponding to this `end` value. |
| int firstSpanForThisEnd = uniqueIdentifierForSpan(0, end); |
| |
| // Offsets are encoded consecutively so we can find the offset by |
| // subtracting: |
| int offset = span - firstSpanForThisEnd; |
| |
| if (offset < 0) { |
| // Oops, `end` must have been too large. Decrement and try again. |
| assert(end > 0); |
| --end; |
| } else if (offset > end) { |
| // Oops, `end` must have been too small. Increment and try again. |
| ++end; |
| } else { |
| return OffsetEndPair(offset, end); |
| } |
| } |
| } |
| |
| /// Combine the given [offset] and [end] into a unique integer that depends |
| /// on both of them, taking advantage of the fact that `0 <= offset <= end`. |
| @visibleForTesting |
| static int uniqueIdentifierForSpan(int offset, int end) { |
| assert(0 <= offset && offset <= end); |
| // Our encoding is based on the observation that if you make a graph of the |
| // set of all possible (offset, end) pairs, marking those that satisfy |
| // `0 <= offset <= end` with an `x`, you get a triangle shape: |
| // |
| // offset |
| // +--------> |
| // |x |
| // |xx |
| // end |xxx |
| // |xxxx |
| // V |
| // |
| // If we assign integers to the `x`s in the order they appear in this graph, |
| // then the rows start with numbers 0, 1, 3, 6, 10, etc. This can be |
| // computed from `end` as `end*(end + 1)/2`. We use `~/` for integer |
| // division. |
| return end * (end + 1) ~/ 2 + offset; |
| } |
| } |
| |
| extension on TypeParameterElement { |
| DartType? get preMigrationBound { |
| var previousElementTypeProvider = ElementTypeProvider.current; |
| try { |
| ElementTypeProvider.current = const ElementTypeProvider(); |
| return bound; |
| } finally { |
| ElementTypeProvider.current = previousElementTypeProvider; |
| } |
| } |
| } |
| |
| extension on FunctionTypedElement { |
| FunctionType get preMigrationType { |
| var previousElementTypeProvider = ElementTypeProvider.current; |
| try { |
| ElementTypeProvider.current = const ElementTypeProvider(); |
| return type; |
| } finally { |
| ElementTypeProvider.current = previousElementTypeProvider; |
| } |
| } |
| } |
| |
| extension on VariableElement { |
| DartType get preMigrationType { |
| var previousElementTypeProvider = ElementTypeProvider.current; |
| try { |
| ElementTypeProvider.current = const ElementTypeProvider(); |
| return type; |
| } finally { |
| ElementTypeProvider.current = previousElementTypeProvider; |
| } |
| } |
| } |
| |
| extension on ExtensionElement { |
| DartType get preMigrationExtendedType { |
| var previousElementTypeProvider = ElementTypeProvider.current; |
| try { |
| ElementTypeProvider.current = const ElementTypeProvider(); |
| return extendedType; |
| } finally { |
| ElementTypeProvider.current = previousElementTypeProvider; |
| } |
| } |
| } |