|
1 | 1 | import 'package:analyzer/dart/analysis/analysis_context.dart'; |
2 | 2 | import 'package:analyzer/dart/analysis/results.dart'; |
| 3 | +import 'package:analyzer/dart/ast/ast.dart'; |
3 | 4 | import 'package:analyzer/dart/constant/value.dart'; |
4 | 5 | import 'package:analyzer/dart/element/element.dart'; |
| 6 | +import 'package:analyzer/dart/element/element2.dart'; |
5 | 7 | import 'package:analyzer/dart/element/type.dart'; |
| 8 | +import 'package:analyzer/src/dart/analysis/session_helper.dart'; |
| 9 | +import 'package:analyzer/src/dart/ast/ast.dart'; |
6 | 10 | import 'package:analyzer/src/dart/resolver/scope.dart'; |
7 | 11 | import 'package:celest_cli/src/analyzer/analysis_error.dart'; |
8 | 12 | import 'package:celest_cli/src/analyzer/resolver/project_resolver.dart'; |
@@ -53,6 +57,11 @@ mixin CelestAnalysisHelpers implements CelestErrorReporter { |
53 | 57 | Set<InterfaceElement> get customExceptionTypes; |
54 | 58 | Set<InterfaceElement> get customModelTypes; |
55 | 59 |
|
| 60 | + static final Expando<AnalysisSessionHelper> _sessionHelpers = Expando(); |
| 61 | + AnalysisSessionHelper get helper => |
| 62 | + _sessionHelpers[context.currentSession] ??= |
| 63 | + AnalysisSessionHelper(context.currentSession); |
| 64 | + |
56 | 65 | final pendingEdits = <String, Set<SourceEdit>>{}; |
57 | 66 |
|
58 | 67 | static final Logger _logger = Logger('CelestAnalyzer'); |
@@ -292,6 +301,84 @@ mixin CelestAnalysisHelpers implements CelestErrorReporter { |
292 | 301 | ); |
293 | 302 | } |
294 | 303 | } |
| 304 | + |
| 305 | + /// Resolves the schema version of a Drift database class. |
| 306 | + Future<int> resolveSchemaVersion(ClassElement2 databaseClass) async { |
| 307 | + final schemaVersionMethod = |
| 308 | + databaseClass.getGetter2('schemaVersion')?.variable3; |
| 309 | + if (schemaVersionMethod == null) { |
| 310 | + reportError( |
| 311 | + 'Invalid database class: No `schemaVersion` getter.', |
| 312 | + severity: AnalysisErrorSeverity.error, |
| 313 | + location: databaseClass.sourceLocation, |
| 314 | + ); |
| 315 | + return -1; |
| 316 | + } |
| 317 | + final declaration = await helper.getElementDeclaration( |
| 318 | + schemaVersionMethod.isSynthetic |
| 319 | + ? schemaVersionMethod.getter2!.firstFragment |
| 320 | + : schemaVersionMethod.firstFragment, |
| 321 | + ); |
| 322 | + |
| 323 | + int? resolveExpression(Expression? expression) { |
| 324 | + switch (expression) { |
| 325 | + // int get schemaVersion => 1; |
| 326 | + case IntegerLiteral(value: final version?): |
| 327 | + return version; |
| 328 | + |
| 329 | + // int get schemaVersion => someConstVariable; |
| 330 | + case Identifier( |
| 331 | + staticElement: PropertyAccessorElement(:final variable2?) |
| 332 | + ): |
| 333 | + if (variable2.computeConstantValue()?.toIntValue() |
| 334 | + case final version?) { |
| 335 | + return version; |
| 336 | + } |
| 337 | + } |
| 338 | + |
| 339 | + return null; |
| 340 | + } |
| 341 | + |
| 342 | + switch (declaration?.node) { |
| 343 | + case VariableDeclaration(:final initializer): |
| 344 | + if (resolveExpression(initializer) case final version?) { |
| 345 | + return version; |
| 346 | + } |
| 347 | + |
| 348 | + // Could not determine schema version. |
| 349 | + reportError( |
| 350 | + 'Invalid `schemaVersion` declaration: Must resolve to an integer.', |
| 351 | + severity: AnalysisErrorSeverity.error, |
| 352 | + location: schemaVersionMethod.sourceLocation, |
| 353 | + ); |
| 354 | + return -1; |
| 355 | + |
| 356 | + case MethodDeclaration(:final body): |
| 357 | + switch (body) { |
| 358 | + case ExpressionFunctionBody(:final expression): |
| 359 | + if (resolveExpression(expression) case final version?) { |
| 360 | + return version; |
| 361 | + } |
| 362 | + case BlockFunctionBody(): |
| 363 | + case EmptyFunctionBody(): |
| 364 | + case NativeFunctionBody(): |
| 365 | + } |
| 366 | + |
| 367 | + // Could not determine schema version. |
| 368 | + reportError( |
| 369 | + 'Invalid `schemaVersion` declaration: Must use => syntax.', |
| 370 | + severity: AnalysisErrorSeverity.error, |
| 371 | + location: schemaVersionMethod.sourceLocation, |
| 372 | + ); |
| 373 | + return -1; |
| 374 | + |
| 375 | + case final unknown: |
| 376 | + throw StateError( |
| 377 | + 'Failed to resolve method declaration for $schemaVersionMethod: ' |
| 378 | + 'Got $unknown (${unknown.runtimeType})', |
| 379 | + ); |
| 380 | + } |
| 381 | + } |
295 | 382 | } |
296 | 383 |
|
297 | 384 | extension TopLevelConstants on LibraryElement { |
|
0 commit comments