blob: 312c6a5b6f0106f87933a0754e8f7610fe79383c [file] [log] [blame]
// Copyright (c) 2023, 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:test/test.dart';
import 'package:test_common/logging.dart';
import 'package:test_common/test_sdk_configuration.dart';
import 'package:vm_service/vm_service.dart';
import '../../fixtures/context.dart';
import '../../fixtures/project.dart';
import '../../fixtures/utilities.dart';
import 'test_inspector.dart';
void runTests({
required TestSdkConfigurationProvider provider,
required CompilationMode compilationMode,
required bool canaryFeatures,
required NullSafety nullSafetyMode,
required bool debug,
}) {
final project = nullSafetyMode == NullSafety.sound
? TestProject.testPackageWithSoundNullSafety()
: TestProject.testPackageWithWeakNullSafety();
final context = TestContext(project, provider);
late VmService service;
late Stream<Event> stream;
late String isolateId;
late ScriptRef mainScript;
final testInspector = TestInspector(context);
onBreakPoint(breakPointId, body) => testInspector.onBreakPoint(
stream,
isolateId,
mainScript,
breakPointId,
body,
);
getInstance(frame, expression) =>
testInspector.getInstance(isolateId, frame, expression);
getObject(instanceId) => service.getObject(isolateId, instanceId);
getInstanceRef(frame, expression) =>
testInspector.getInstanceRef(isolateId, frame, expression);
getFields(instanceRef, {offset, count}) => testInspector
.getFields(isolateId, instanceRef, offset: offset, count: count);
group('$nullSafetyMode |', () {
group('$compilationMode |', () {
setUpAll(() async {
setCurrentLogWriter(debug: debug);
await context.setUp(
testSettings: TestSettings(
compilationMode: compilationMode,
enableExpressionEvaluation: true,
verboseCompiler: debug,
canaryFeatures: canaryFeatures,
experiments: ['records'],
),
);
service = context.debugConnection.vmService;
final vm = await service.getVM();
isolateId = vm.isolates!.first.id!;
final scripts = await service.getScripts(isolateId);
await service.streamListen('Debug');
stream = service.onEvent('Debug');
mainScript = scripts.scripts!
.firstWhere((each) => each.uri!.contains('main.dart'));
});
tearDownAll(context.tearDown);
setUp(() => setCurrentLogWriter(debug: debug));
tearDown(() async {
try {
await service.resume(isolateId);
} catch (_) {}
});
group('Library |', () {
test('classes', () async {
const libraryId = 'org-dartlang-app:///web/main.dart';
final library = await getObject(libraryId);
expect(
library,
isA<Library>().having(
(l) => l.classes,
'classes',
[
matchClassRef(name: 'MainClass', libraryId: libraryId),
matchClassRef(name: 'EnclosedClass', libraryId: libraryId),
matchClassRef(name: 'ClassWithMethod', libraryId: libraryId),
matchClassRef(name: 'EnclosingClass', libraryId: libraryId),
],
),
);
});
});
group('Class |', () {
test('name and library', () async {
const libraryId = 'org-dartlang-app:///web/main.dart';
const className = 'MainClass';
final cls = await getObject('classes|$libraryId|$className');
expect(cls, matchClass(name: className, libraryId: libraryId));
});
});
group('Object |', () {
test('type and fields', () async {
await onBreakPoint('printFieldMain', (event) async {
final frame = event.topFrame!.index!;
final instanceRef = await getInstanceRef(frame, 'instance');
final instanceId = instanceRef.id!;
expect(
await getObject(instanceId),
matchPlainInstance(
libraryId: 'org-dartlang-app:///web/main.dart',
type: 'MainClass',
),
);
expect(await getFields(instanceRef), {'_field': 1, 'field': 2});
// Offsets and counts are ignored for plain object fields.
// DevTools calls [VmServiceInterface.getObject] with offset=0
// and count=0 and expects all fields to be returned.
expect(
await getFields(instanceRef, offset: 0, count: 0),
{'_field': 1, 'field': 2},
);
expect(
await getFields(instanceRef, offset: 0),
{'_field': 1, 'field': 2},
);
expect(
await getFields(instanceRef, offset: 0, count: 1),
{'_field': 1, 'field': 2},
);
expect(
await getFields(instanceRef, offset: 1),
{'_field': 1, 'field': 2},
);
expect(
await getFields(instanceRef, offset: 1, count: 0),
{'_field': 1, 'field': 2},
);
expect(
await getFields(instanceRef, offset: 1, count: 3),
{'_field': 1, 'field': 2},
);
});
});
test('field access', () async {
await onBreakPoint('printFieldMain', (event) async {
final frame = event.topFrame!.index!;
expect(
await getInstance(frame, r'instance.field'),
matchPrimitiveInstance(kind: InstanceKind.kDouble, value: 2),
);
expect(
await getInstance(frame, r'instance._field'),
matchPrimitiveInstance(kind: InstanceKind.kDouble, value: 1),
);
});
});
});
group('List |', () {
test('type and fields', () async {
await onBreakPoint('printList', (event) async {
final frame = event.topFrame!.index!;
final instanceRef = await getInstanceRef(frame, 'list');
final instanceId = instanceRef.id!;
expect(await getObject(instanceId), matchListInstance(type: 'int'));
expect(
await getFields(instanceRef),
{0: 0.0, 1: 1.0, 2: 2.0},
);
expect(await getFields(instanceRef, offset: 1, count: 0), {});
expect(
await getFields(instanceRef, offset: 0),
{0: 0.0, 1: 1.0, 2: 2.0},
);
expect(
await getFields(instanceRef, offset: 0, count: 1),
{0: 0.0},
);
expect(
await getFields(instanceRef, offset: 1),
{0: 1.0, 1: 2.0},
);
expect(
await getFields(instanceRef, offset: 1, count: 1),
{0: 1.0},
);
expect(
await getFields(instanceRef, offset: 1, count: 3),
{0: 1.0, 1: 2.0},
);
expect(await getFields(instanceRef, offset: 3, count: 3), {});
});
});
test('Element access', () async {
await onBreakPoint('printList', (event) async {
final frame = event.topFrame!.index!;
expect(
await getInstance(frame, r'list[0]'),
matchPrimitiveInstance(kind: InstanceKind.kDouble, value: 0),
);
expect(
await getInstance(frame, r"list[1]"),
matchPrimitiveInstance(kind: InstanceKind.kDouble, value: 1),
);
expect(
await getInstance(frame, r"list[2]"),
matchPrimitiveInstance(kind: InstanceKind.kDouble, value: 2),
);
});
});
});
group('Map |', () {
test('type and fields', () async {
await onBreakPoint('printMap', (event) async {
final frame = event.topFrame!.index!;
final instanceRef = await getInstanceRef(frame, 'map');
final instanceId = instanceRef.id!;
expect(
await getObject(instanceId),
matchMapInstance(type: 'IdentityMap<String, int>'),
);
expect(await getFields(instanceRef), {'a': 1, 'b': 2, 'c': 3});
expect(await getFields(instanceRef, offset: 1, count: 0), {});
expect(
await getFields(instanceRef, offset: 0),
{'a': 1, 'b': 2, 'c': 3},
);
expect(await getFields(instanceRef, offset: 0, count: 1), {'a': 1});
expect(await getFields(instanceRef, offset: 1), {'b': 2, 'c': 3});
expect(await getFields(instanceRef, offset: 1, count: 1), {'b': 2});
expect(
await getFields(instanceRef, offset: 1, count: 3),
{'b': 2, 'c': 3},
);
expect(await getFields(instanceRef, offset: 3, count: 3), {});
});
});
test('Element access', () async {
await onBreakPoint('printMap', (event) async {
final frame = event.topFrame!.index!;
expect(
await getInstance(frame, r"map['a']"),
matchPrimitiveInstance(kind: InstanceKind.kDouble, value: 1),
);
expect(
await getInstance(frame, r"map['b']"),
matchPrimitiveInstance(kind: InstanceKind.kDouble, value: 2),
);
expect(
await getInstance(frame, r"map['c']"),
matchPrimitiveInstance(kind: InstanceKind.kDouble, value: 3),
);
});
});
});
group('Set |', () {
test('type and fields', () async {
await onBreakPoint('printSet', (event) async {
final frame = event.topFrame!.index!;
final instanceRef = await getInstanceRef(frame, 'mySet');
final instanceId = instanceRef.id!;
expect(
await getObject(instanceId),
matchSetInstance(type: '_HashSet<int>'),
);
expect(
await getFields(instanceRef),
{0: 1.0, 1: 4.0, 2: 5.0, 3: 7.0},
);
expect(
await getFields(instanceRef, offset: 0),
{0: 1.0, 1: 4.0, 2: 5.0, 3: 7.0},
);
expect(
await getFields(instanceRef, offset: 1, count: 2),
{0: 4.0, 1: 5.0},
);
expect(
await getFields(instanceRef, offset: 2),
{0: 5.0, 1: 7.0},
);
expect(
await getFields(instanceRef, offset: 2, count: 10),
{0: 5.0, 1: 7.0},
);
expect(await getFields(instanceRef, offset: 1, count: 0), {});
expect(await getFields(instanceRef, offset: 10, count: 2), {});
});
});
test('Element access', () async {
await onBreakPoint('printSet', (event) async {
final frame = event.topFrame!.index!;
expect(
await getInstance(frame, r"mySet.first"),
matchPrimitiveInstance(kind: InstanceKind.kDouble, value: 1),
);
expect(
await getInstance(frame, r"mySet.last"),
matchPrimitiveInstance(kind: InstanceKind.kDouble, value: 7),
);
});
});
});
});
});
}