mirror of https://github.com/AMT-Cheif/drift.git
150 lines
4.9 KiB
Dart
150 lines
4.9 KiB
Dart
import 'dart:io';
|
|
import 'dart:isolate';
|
|
|
|
import 'package:analyzer/dart/analysis/analysis_context.dart';
|
|
import 'package:analyzer/dart/analysis/analysis_context_collection.dart';
|
|
import 'package:analyzer/dart/analysis/results.dart';
|
|
import 'package:analyzer/dart/ast/ast.dart';
|
|
import 'package:analyzer/dart/element/element.dart';
|
|
import 'package:analyzer/file_system/overlay_file_system.dart';
|
|
import 'package:analyzer/file_system/physical_file_system.dart';
|
|
import 'package:build/build.dart';
|
|
import 'package:drift_dev/src/analysis/backend.dart';
|
|
import 'package:drift_dev/src/analysis/driver/driver.dart';
|
|
import 'package:drift_dev/src/analysis/driver/error.dart';
|
|
import 'package:drift_dev/src/analysis/driver/state.dart';
|
|
import 'package:drift_dev/src/analyzer/options.dart';
|
|
import 'package:logging/logging.dart';
|
|
import 'package:package_config/package_config.dart';
|
|
import 'package:path/path.dart' as p;
|
|
import 'package:test/test.dart';
|
|
|
|
/// A [DriftBackend] implementation used for testing.
|
|
///
|
|
/// This backend has limited support for Dart analysis: [sourceContents] forming
|
|
/// the package `a` are available for analysis. In addition, `drift` and
|
|
/// `drift_dev` imports can be analyzed as well.
|
|
class TestBackend extends DriftBackend {
|
|
final Map<String, String> sourceContents;
|
|
|
|
late final DriftAnalysisDriver driver;
|
|
|
|
AnalysisContext? _dartContext;
|
|
|
|
TestBackend(Map<String, String> sourceContents,
|
|
{DriftOptions options = const DriftOptions.defaults()})
|
|
: sourceContents = {
|
|
for (final entry in sourceContents.entries)
|
|
AssetId.parse(entry.key).uri.toString(): entry.value,
|
|
} {
|
|
driver = DriftAnalysisDriver(this, options);
|
|
}
|
|
|
|
factory TestBackend.inTest(Map<String, String> sourceContents,
|
|
{DriftOptions options = const DriftOptions.defaults()}) {
|
|
final backend = TestBackend(sourceContents, options: options);
|
|
addTearDown(backend.dispose);
|
|
|
|
return backend;
|
|
}
|
|
|
|
Future<void> _setupDartAnalyzer() async {
|
|
final provider = OverlayResourceProvider(PhysicalResourceProvider.INSTANCE);
|
|
|
|
// Analyze example sources against the drift sources from the current
|
|
// drift_dev test runner.
|
|
final uri = await Isolate.packageConfig;
|
|
final hostConfig =
|
|
PackageConfig.parseBytes(await File.fromUri(uri!).readAsBytes(), uri);
|
|
final testConfig = PackageConfig([
|
|
...hostConfig.packages,
|
|
Package('a', Uri.directory('/a/'), packageUriRoot: Uri.parse('lib/')),
|
|
]);
|
|
|
|
// Write package config used to analyze dummy sources
|
|
final configBuffer = StringBuffer();
|
|
PackageConfig.writeString(testConfig, configBuffer);
|
|
provider.setOverlay('/a/.dart_tool/package_config.json',
|
|
content: configBuffer.toString(), modificationStamp: 1);
|
|
|
|
// Also put sources into the overlay:
|
|
sourceContents.forEach((key, value) {
|
|
final uri = Uri.parse(key);
|
|
|
|
if (uri.scheme == 'package') {
|
|
final package = uri.pathSegments.first;
|
|
final path =
|
|
p.url.joinAll(['/$package/lib', ...uri.pathSegments.skip(1)]);
|
|
|
|
provider.setOverlay(path, content: value, modificationStamp: 1);
|
|
}
|
|
});
|
|
|
|
final collection = AnalysisContextCollection(
|
|
includedPaths: ['/a'],
|
|
resourceProvider: provider,
|
|
);
|
|
|
|
_dartContext = collection.contexts.single;
|
|
}
|
|
|
|
Future<void> ensureHasDartAnalyzer() async {
|
|
if (_dartContext == null) {
|
|
await _setupDartAnalyzer();
|
|
}
|
|
}
|
|
|
|
@override
|
|
Logger get log => Logger.root;
|
|
|
|
@override
|
|
Uri resolveUri(Uri base, String uriString) {
|
|
return base.resolve(uriString);
|
|
}
|
|
|
|
@override
|
|
Future<String> readAsString(Uri uri) async {
|
|
return sourceContents[uri.toString()] ??
|
|
(throw StateError('No source for $uri'));
|
|
}
|
|
|
|
@override
|
|
Future<LibraryElement> readDart(Uri uri) async {
|
|
await ensureHasDartAnalyzer();
|
|
final result =
|
|
await _dartContext!.currentSession.getLibraryByUri(uri.toString());
|
|
|
|
return (result as LibraryElementResult).element;
|
|
}
|
|
|
|
@override
|
|
Future<AstNode?> loadElementDeclaration(Element element) async {
|
|
final library = element.library;
|
|
if (library == null) return null;
|
|
|
|
final info = await library.session.getResolvedLibraryByElement(library);
|
|
if (info is ResolvedLibraryResult) {
|
|
return info.getElementDeclaration(element)?.node;
|
|
} else {
|
|
return null;
|
|
}
|
|
}
|
|
|
|
Future<void> dispose() async {}
|
|
}
|
|
|
|
Matcher get hasNoErrors => isA<FileState>()
|
|
.having((e) => e.errorsDuringDiscovery, 'errorsDuringDiscovery', isEmpty)
|
|
.having((e) => e.analysis.values.expand((e) => e.errorsDuringAnalysis),
|
|
'(errors in analyzed elements)', isEmpty);
|
|
|
|
TypeMatcher<DriftAnalysisError> isDriftError(dynamic message) {
|
|
return isA<DriftAnalysisError>().having((e) => e.message, 'message', message);
|
|
}
|
|
|
|
extension DriftErrorMatchers on TypeMatcher<DriftAnalysisError> {
|
|
TypeMatcher<DriftAnalysisError> withSpan(lexemeMatcher) {
|
|
return having((e) => e.span?.text, 'span.text', lexemeMatcher);
|
|
}
|
|
}
|