// Copyright 2014 The Flutter Authors. 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:convert' show jsonEncode; import 'dart:io' show Directory, File; import 'package:coverage/src/hitmap.dart'; import 'package:flutter_tools/src/test/coverage_collector.dart'; import 'package:flutter_tools/src/test/test_device.dart' show TestDevice; import 'package:stream_channel/stream_channel.dart' show StreamChannel; import 'package:vm_service/vm_service.dart'; import '../src/common.dart'; import '../src/fake_vm_services.dart'; void main() { testWithoutContext('Coverage collector Can handle coverage SentinelException', () async { final FakeVmServiceHost fakeVmServiceHost = FakeVmServiceHost( requests: [ FakeVmServiceRequest( method: 'getVersion', jsonResponse: Version(major: 3, minor: 51).toJson(), ), FakeVmServiceRequest( method: 'getVM', jsonResponse: (VM.parse({})! ..isolates = [ IsolateRef.parse({ 'id': '1', })!, ] ).toJson(), ), const FakeVmServiceRequest( method: 'getScripts', args: { 'isolateId': '1', }, jsonResponse: { 'type': 'Sentinel', }, ), ], ); final Map result = await collect( null, {'foo'}, connector: (Uri? uri) async { return fakeVmServiceHost.vmService; }, ); expect(result, {'type': 'CodeCoverage', 'coverage': []}); expect(fakeVmServiceHost.hasRemainingExpectations, false); }); testWithoutContext('Coverage collector processes coverage and script data', () async { final FakeVmServiceHost fakeVmServiceHost = FakeVmServiceHost( requests: [ FakeVmServiceRequest( method: 'getVersion', jsonResponse: Version(major: 3, minor: 51).toJson(), ), FakeVmServiceRequest( method: 'getVM', jsonResponse: (VM.parse({})! ..isolates = [ IsolateRef.parse({ 'id': '1', })!, ] ).toJson(), ), FakeVmServiceRequest( method: 'getScripts', args: { 'isolateId': '1', }, jsonResponse: ScriptList(scripts: [ ScriptRef(uri: 'package:foo/foo.dart', id: '1'), ScriptRef(uri: 'package:bar/bar.dart', id: '2'), ]).toJson(), ), FakeVmServiceRequest( method: 'getSourceReport', args: { 'isolateId': '1', 'reports': ['Coverage'], 'scriptId': '1', 'forceCompile': true, 'reportLines': true, }, jsonResponse: SourceReport( ranges: [ SourceReportRange( scriptIndex: 0, startPos: 0, endPos: 0, compiled: true, coverage: SourceReportCoverage( hits: [1, 3], misses: [2], ), ), ], scripts: [ ScriptRef( uri: 'package:foo/foo.dart', id: '1', ), ], ).toJson(), ), ], ); final Map result = await collect( null, {'foo'}, connector: (Uri? uri) async { return fakeVmServiceHost.vmService; }, ); expect(result, { 'type': 'CodeCoverage', 'coverage': [ { 'source': 'package:foo/foo.dart', 'script': { 'type': '@Script', 'fixedId': true, 'id': 'libraries/1/scripts/package%3Afoo%2Ffoo.dart', 'uri': 'package:foo/foo.dart', '_kind': 'library', }, 'hits': [1, 1, 3, 1, 2, 0], }, ], }); expect(fakeVmServiceHost.hasRemainingExpectations, false); }); testWithoutContext('Coverage collector with null libraryNames accepts all libraries', () async { final FakeVmServiceHost fakeVmServiceHost = createFakeVmServiceHostWithFooAndBar(); final Map result = await collect( null, null, connector: (Uri? uri) async { return fakeVmServiceHost.vmService; }, ); expect(result, { 'type': 'CodeCoverage', 'coverage': [ { 'source': 'package:foo/foo.dart', 'script': { 'type': '@Script', 'fixedId': true, 'id': 'libraries/1/scripts/package%3Afoo%2Ffoo.dart', 'uri': 'package:foo/foo.dart', '_kind': 'library', }, 'hits': [1, 1, 3, 1, 2, 0], }, { 'source': 'package:bar/bar.dart', 'script': { 'type': '@Script', 'fixedId': true, 'id': 'libraries/1/scripts/package%3Abar%2Fbar.dart', 'uri': 'package:bar/bar.dart', '_kind': 'library', }, 'hits': [47, 1, 21, 1, 32, 0, 86, 0], }, ], }); expect(fakeVmServiceHost.hasRemainingExpectations, false); }); testWithoutContext('Coverage collector with libraryFilters', () async { final FakeVmServiceHost fakeVmServiceHost = FakeVmServiceHost( requests: [ FakeVmServiceRequest( method: 'getVersion', jsonResponse: Version(major: 3, minor: 57).toJson(), ), FakeVmServiceRequest( method: 'getVM', jsonResponse: (VM.parse({})! ..isolates = [ IsolateRef.parse({ 'id': '1', })!, ] ).toJson(), ), FakeVmServiceRequest( method: 'getSourceReport', args: { 'isolateId': '1', 'reports': ['Coverage'], 'forceCompile': true, 'reportLines': true, 'libraryFilters': ['package:foo/'], }, jsonResponse: SourceReport( ranges: [ SourceReportRange( scriptIndex: 0, startPos: 0, endPos: 0, compiled: true, coverage: SourceReportCoverage( hits: [1, 3], misses: [2], ), ), ], scripts: [ ScriptRef( uri: 'package:foo/foo.dart', id: '1', ), ], ).toJson(), ), ], ); final Map result = await collect( null, {'foo'}, connector: (Uri? uri) async { return fakeVmServiceHost.vmService; }, ); expect(result, { 'type': 'CodeCoverage', 'coverage': [ { 'source': 'package:foo/foo.dart', 'script': { 'type': '@Script', 'fixedId': true, 'id': 'libraries/1/scripts/package%3Afoo%2Ffoo.dart', 'uri': 'package:foo/foo.dart', '_kind': 'library', }, 'hits': [1, 1, 3, 1, 2, 0], }, ], }); expect(fakeVmServiceHost.hasRemainingExpectations, false); }); testWithoutContext('Coverage collector with libraryFilters and null libraryNames', () async { final FakeVmServiceHost fakeVmServiceHost = FakeVmServiceHost( requests: [ FakeVmServiceRequest( method: 'getVersion', jsonResponse: Version(major: 3, minor: 57).toJson(), ), FakeVmServiceRequest( method: 'getVM', jsonResponse: (VM.parse({})! ..isolates = [ IsolateRef.parse({ 'id': '1', })!, ] ).toJson(), ), FakeVmServiceRequest( method: 'getSourceReport', args: { 'isolateId': '1', 'reports': ['Coverage'], 'forceCompile': true, 'reportLines': true, }, jsonResponse: SourceReport( ranges: [ SourceReportRange( scriptIndex: 0, startPos: 0, endPos: 0, compiled: true, coverage: SourceReportCoverage( hits: [1, 3], misses: [2], ), ), ], scripts: [ ScriptRef( uri: 'package:foo/foo.dart', id: '1', ), ], ).toJson(), ), ], ); final Map result = await collect( null, null, connector: (Uri? uri) async { return fakeVmServiceHost.vmService; }, ); expect(result, { 'type': 'CodeCoverage', 'coverage': [ { 'source': 'package:foo/foo.dart', 'script': { 'type': '@Script', 'fixedId': true, 'id': 'libraries/1/scripts/package%3Afoo%2Ffoo.dart', 'uri': 'package:foo/foo.dart', '_kind': 'library', }, 'hits': [1, 1, 3, 1, 2, 0], }, ], }); expect(fakeVmServiceHost.hasRemainingExpectations, false); }); testWithoutContext('Coverage collector caches read files', () async { Directory? tempDir; try { tempDir = Directory.systemTemp.createTempSync('flutter_coverage_collector_test.'); final File file = File('${tempDir.path}/packages.json'); file.writeAsStringSync(jsonEncode({ 'configVersion': 2, 'packages': >[ { 'name': 'foo', 'rootUri': 'foo', }, { 'name': 'bar', 'rootUri': 'bar', }, ], })); final Directory fooDir = Directory('${tempDir.path}/foo/'); fooDir.createSync(); final File fooFile = File('${fooDir.path}/foo.dart'); fooFile.writeAsStringSync('hit\nnohit but ignored // coverage:ignore-line\nhit\n'); final String packagesPath = file.path; final CoverageCollector collector = CoverageCollector( libraryNames: {'foo', 'bar'}, verbose: false, packagesPath: packagesPath, resolver: await CoverageCollector.getResolver(packagesPath) ); await collector.collectCoverage(TestTestDevice(), connector: (Uri? uri) async { return createFakeVmServiceHostWithFooAndBar().vmService; }); Future getHitMapAndVerify() async { final Map gottenHitmap = {}; await collector.finalizeCoverage(formatter: (Map hitmap) { gottenHitmap.addAll(hitmap); return ''; }); expect(gottenHitmap.keys.toList()..sort(), ['package:bar/bar.dart', 'package:foo/foo.dart']); expect(gottenHitmap['package:foo/foo.dart']!.lineHits, {1: 1, /* 2: 0, is ignored in file */ 3: 1}); expect(gottenHitmap['package:bar/bar.dart']!.lineHits, {21: 1, 32: 0, 47: 1, 86: 0}); } Future verifyHitmapEmpty() async { final Map gottenHitmap = {}; await collector.finalizeCoverage(formatter: (Map hitmap) { gottenHitmap.addAll(hitmap); return ''; }); expect(gottenHitmap.isEmpty, isTrue); } // Get hit map the first time. await getHitMapAndVerify(); // Getting the hitmap clears it so we now doesn't get any data. await verifyHitmapEmpty(); // Collecting again gets us the same data even though the foo file has been deleted. // This means that the fact that line 2 was ignored has been cached. fooFile.deleteSync(); await collector.collectCoverage(TestTestDevice(), connector: (Uri? uri) async { return createFakeVmServiceHostWithFooAndBar().vmService; }); await getHitMapAndVerify(); } finally { tempDir?.deleteSync(recursive: true); } }); } FakeVmServiceHost createFakeVmServiceHostWithFooAndBar() { return FakeVmServiceHost( requests: [ FakeVmServiceRequest( method: 'getVersion', jsonResponse: Version(major: 3, minor: 51).toJson(), ), FakeVmServiceRequest( method: 'getVM', jsonResponse: (VM.parse({})! ..isolates = [ IsolateRef.parse({ 'id': '1', })!, ] ).toJson(), ), FakeVmServiceRequest( method: 'getScripts', args: { 'isolateId': '1', }, jsonResponse: ScriptList(scripts: [ ScriptRef(uri: 'package:foo/foo.dart', id: '1'), ScriptRef(uri: 'package:bar/bar.dart', id: '2'), ]).toJson(), ), FakeVmServiceRequest( method: 'getSourceReport', args: { 'isolateId': '1', 'reports': ['Coverage'], 'scriptId': '1', 'forceCompile': true, 'reportLines': true, }, jsonResponse: SourceReport( ranges: [ SourceReportRange( scriptIndex: 0, startPos: 0, endPos: 0, compiled: true, coverage: SourceReportCoverage( hits: [1, 3], misses: [2], ), ), ], scripts: [ ScriptRef( uri: 'package:foo/foo.dart', id: '1', ), ], ).toJson(), ), FakeVmServiceRequest( method: 'getSourceReport', args: { 'isolateId': '1', 'reports': ['Coverage'], 'scriptId': '2', 'forceCompile': true, 'reportLines': true, }, jsonResponse: SourceReport( ranges: [ SourceReportRange( scriptIndex: 0, startPos: 0, endPos: 0, compiled: true, coverage: SourceReportCoverage( hits: [47, 21], misses: [32, 86], ), ), ], scripts: [ ScriptRef( uri: 'package:bar/bar.dart', id: '2', ), ], ).toJson(), ), ], ); } class TestTestDevice extends TestDevice { @override Future get finished => Future.delayed(const Duration(seconds: 1)); @override Future kill() => Future.value(); @override Future get observatoryUri => Future.value(); @override Future> start(String entrypointPath) { throw UnimplementedError(); } }