Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Code coverage #2294

Merged
merged 31 commits into from
Sep 13, 2024
Merged
Show file tree
Hide file tree
Changes from 4 commits
Commits
Show all changes
31 commits
Select commit Hold shift + click to select a range
4ce06bd
Basic coverage
gogolon Jun 6, 2024
9639f11
Use new coverage version, exclude generated files, fix bug causing pa…
gogolon Aug 2, 2024
e1280ee
WIP pause isolates on exit
gogolon Aug 6, 2024
54bc5af
Collect coverage from paused isolates
gogolon Aug 6, 2024
d7634d0
Use rxdart, split coverage related code into files
gogolon Aug 17, 2024
dd48f98
Use wrapper for port forwarding
gogolon Aug 17, 2024
d4a2722
Fix disposing serviceClient, unify named params usage
gogolon Aug 17, 2024
3ee5bce
Rename stopped => coverageCollected
gogolon Aug 17, 2024
143d85e
Only collect coverage from paused isolates if event type is pauseExit
gogolon Aug 17, 2024
8a2e6c7
Do not collect coverage from isolates to avoid duplicate entries
gogolon Aug 22, 2024
1673280
Add doc comment about the host port
gogolon Aug 22, 2024
d6c6945
Use named import
gogolon Aug 22, 2024
b79bc9c
Avoid dart:io platform, refactor & test VMConnectionDetails, commas
gogolon Aug 27, 2024
faee594
Remove rxdart
gogolon Aug 27, 2024
c045089
Use completer
gogolon Aug 29, 2024
f6394cd
Automatically find and forward an unused port instead of using a hard…
gogolon Sep 8, 2024
73053cd
Merge with main
gogolon Sep 9, 2024
96b9d07
Use adb 0.4.0
gogolon Sep 11, 2024
5774f60
Bump patrol & patrol_cli version, fill compatibility map in Compatibi…
gogolon Sep 11, 2024
0b1d919
Update table in docs
gogolon Sep 11, 2024
4e3c740
Swap columns in the docs table
gogolon Sep 12, 2024
c8786ce
Rework VesionComparator, write tests
gogolon Sep 12, 2024
c038ad0
Fix version comparator
gogolon Sep 12, 2024
bcf3a7c
Extract VersionComparator to a separate file
gogolon Sep 12, 2024
3026f87
Add missing import
gogolon Sep 12, 2024
c0cbf9b
Rename field, change comment
gogolon Sep 12, 2024
351cf7b
Remove redundant code
gogolon Sep 12, 2024
dd11d0a
Update changelog and docs
gogolon Sep 12, 2024
c303877
Merge remote-tracking branch 'upstream/master' into features/code-cov…
gogolon Sep 13, 2024
4c905f6
Fix changelog
gogolon Sep 13, 2024
569dc8c
Add empty line to CLI changelog
gogolon Sep 13, 2024
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
24 changes: 24 additions & 0 deletions packages/patrol/lib/src/binding.dart
Original file line number Diff line number Diff line change
@@ -1,5 +1,7 @@
import 'dart:convert';
import 'dart:developer';
import 'dart:io' as io;
import 'dart:isolate';

import 'package:flutter/foundation.dart';
import 'package:flutter/material.dart';
Expand Down Expand Up @@ -93,6 +95,27 @@ class PatrolBinding extends LiveTestWidgetsFlutterBinding {
final nameOfRequestedTest = await patrolAppService.testExecutionRequested;

if (nameOfRequestedTest == _currentDartTest) {
if (const bool.fromEnvironment('COVERAGE_ENABLED')) {
postEvent(
'waitForCoverageCollection',
{'mainIsolateId': Service.getIsolateId(Isolate.current)},
);

var stopped = true;
gogolon marked this conversation as resolved.
Show resolved Hide resolved

registerExtension('ext.patrol.markTestCompleted',
(method, parameters) async {
stopped = false;
return ServiceExtensionResponse.result(jsonEncode({}));
});

while (stopped) {
// The loop is needed to keep this isolate alive until the coverage
// data is collected.
await Future<void>.delayed(const Duration(seconds: 1));
}
}

logger(
'finished test $_currentDartTest. Will report its status back to the native side',
);
Expand All @@ -101,6 +124,7 @@ class PatrolBinding extends LiveTestWidgetsFlutterBinding {
logger(
'tearDown(): test "$testName" in group "$_currentDartTest", passed: $passed',
);

await patrolAppService.markDartTestAsCompleted(
dartFileName: _currentDartTest!,
passed: passed,
Expand Down
24 changes: 20 additions & 4 deletions packages/patrol/lib/src/common.dart
Original file line number Diff line number Diff line change
@@ -1,3 +1,4 @@
import 'dart:developer';
import 'dart:io' as io;

import 'package:flutter/foundation.dart';
Expand Down Expand Up @@ -237,19 +238,34 @@ String deduplicateGroupEntryName(String parentName, String currentName) {
);
}

/// Recursively prints the structure of the test suite.
/// Recursively prints the structure of the test suite and reports test count
/// of the top-most group
@internal
void printGroupStructure(DartGroupEntry group, {int indentation = 0}) {
int reportGroupStructure(DartGroupEntry group, {int indentation = 0}) {
gogolon marked this conversation as resolved.
Show resolved Hide resolved
var testCount = group.type == GroupEntryType.test ? 1 : 0;

final indent = ' ' * indentation;
debugPrint("$indent-- group: '${group.name}'");
final tag = group.type == GroupEntryType.group ? 'group' : 'test';
debugPrint("$indent-- $tag: '${group.name}'");

for (final entry in group.entries) {
if (entry.type == GroupEntryType.test) {
++testCount;
debugPrint("$indent -- test: '${entry.name}'");
} else {
for (final subgroup in entry.entries) {
printGroupStructure(subgroup, indentation: indentation + 5);
testCount +=
reportGroupStructure(subgroup, indentation: indentation + 5);
}
}
}

if (indentation == 0) {
postEvent(
'testCount',
{'testCount': testCount},
);
}

return testCount;
}
33 changes: 33 additions & 0 deletions packages/patrol_cli/lib/src/commands/test.dart
Original file line number Diff line number Diff line change
@@ -1,10 +1,13 @@
import 'dart:async';

import 'package:file/file.dart';
import 'package:glob/glob.dart';
import 'package:patrol_cli/src/analytics/analytics.dart';
import 'package:patrol_cli/src/android/android_test_backend.dart';
import 'package:patrol_cli/src/base/extensions/core.dart';
import 'package:patrol_cli/src/base/logger.dart';
import 'package:patrol_cli/src/compatibility_checker.dart';
import 'package:patrol_cli/src/coverage/run_code_coverage.dart';
import 'package:patrol_cli/src/crossplatform/app_options.dart';
import 'package:patrol_cli/src/dart_defines_reader.dart';
import 'package:patrol_cli/src/devices.dart';
Expand All @@ -26,6 +29,7 @@ class TestCommand extends PatrolCommand {
required AndroidTestBackend androidTestBackend,
required IOSTestBackend iosTestBackend,
required MacOSTestBackend macOSTestBackend,
required Directory packageDirectory,
required Analytics analytics,
required Logger logger,
}) : _deviceFinder = deviceFinder,
Expand All @@ -37,6 +41,7 @@ class TestCommand extends PatrolCommand {
_androidTestBackend = androidTestBackend,
_iosTestBackend = iosTestBackend,
_macosTestBackend = macOSTestBackend,
_packageDirectory = packageDirectory,
_analytics = analytics,
_logger = logger {
usesTargetOption();
Expand All @@ -47,6 +52,7 @@ class TestCommand extends PatrolCommand {
usesLabelOption();
usesWaitOption();
usesPortOptions();
useCoverageOptions();

usesUninstallOption();

Expand All @@ -63,6 +69,7 @@ class TestCommand extends PatrolCommand {
final AndroidTestBackend _androidTestBackend;
final IOSTestBackend _iosTestBackend;
final MacOSTestBackend _macosTestBackend;
final Directory _packageDirectory;

final Analytics _analytics;
final Logger _logger;
Expand Down Expand Up @@ -146,6 +153,8 @@ See https://github.com/leancodepl/patrol/issues/1316 to learn more.
final wait = intArg('wait') ?? defaultWait;
final displayLabel = boolArg('label');
final uninstall = boolArg('uninstall');
final coverageEnabled = boolArg('coverage');
final ignoreGlobs = stringsArg('coverage-ignore').map(Glob.new).toSet();

final customDartDefines = {
..._dartDefinesReader.fromFile(),
Expand All @@ -162,6 +171,7 @@ See https://github.com/leancodepl/patrol/issues/1316 to learn more.
'PATROL_TEST_LABEL_ENABLED': displayLabel.toString(),
'PATROL_TEST_SERVER_PORT': super.testServerPort.toString(),
'PATROL_APP_SERVER_PORT': super.appServerPort.toString(),
'COVERAGE_ENABLED': coverageEnabled.toString(),
}.withNullsRemoved();

final dartDefines = {...customDartDefines, ...internalDartDefines};
Expand Down Expand Up @@ -214,6 +224,17 @@ See https://github.com/leancodepl/patrol/issues/1316 to learn more.

await _build(androidOpts, iosOpts, macosOpts, device);
await _preExecute(androidOpts, iosOpts, macosOpts, device, uninstall);

if (coverageEnabled) {
await runCodeCoverage(
flutterPackageName: config.flutterPackageName,
flutterPackageDirectory: _packageDirectory,
platform: device.targetPlatform,
logger: _logger,
ignoreGlobs: ignoreGlobs,
);
}

final allPassed = await _execute(
flutterOpts,
androidOpts,
Expand Down Expand Up @@ -340,4 +361,16 @@ See https://github.com/leancodepl/patrol/issues/1316 to learn more.

return allPassed;
}

void useCoverageOptions() {
argParser
..addFlag(
'coverage',
help: 'Generate coverage.',
)
..addMultiOption(
'coverage-ignore',
help: 'Exclude files from coverage using glob patterns.',
);
}
}
190 changes: 190 additions & 0 deletions packages/patrol_cli/lib/src/coverage/run_code_coverage.dart
Original file line number Diff line number Diff line change
@@ -0,0 +1,190 @@
import 'dart:async';
import 'dart:convert';
import 'dart:io';
gogolon marked this conversation as resolved.
Show resolved Hide resolved

import 'package:coverage/coverage.dart';
import 'package:glob/glob.dart';
import 'package:patrol_cli/src/base/logger.dart';
import 'package:patrol_cli/src/devices.dart';
import 'package:vm_service/vm_service.dart';
import 'package:vm_service/vm_service_io.dart';

Future<Map<String, HitMap>> _collectCoverage(
VmService client,
Uri vmUri,
String packageName,
String mainIsolateId,
) async {
final coverage = await collect(
gogolon marked this conversation as resolved.
Show resolved Hide resolved
vmUri,
false,
false,
false,
{packageName},
);

final socket = await WebSocket.connect(client.wsUri!)
..add(
jsonEncode(
{
'jsonrpc': '2.0',
'id': 21,
'method': 'ext.patrol.markTestCompleted',
'params': {
'isolateId': mainIsolateId,
'command': 'markTestCompleted',
},
},
),
);
await socket.close();

final map = await HitMap.parseJson(
coverage['coverage'] as List<Map<String, dynamic>>,
);

return map;
}

Future<ProcessResult> _forwardAdbPort(String host, String guest) async {
return Process.run('adb', ['forward', 'tcp:$host', 'tcp:$guest']);
gogolon marked this conversation as resolved.
Show resolved Hide resolved
}

Uri _createWebSocketUri(Uri uri) {
final pathSegments = uri.pathSegments.where((c) => c.isNotEmpty).toList()
..add('ws');
return uri.replace(scheme: 'ws', pathSegments: pathSegments);
}

Future<void> _saveCoverage(String report) async {
final coverageDirectory = Directory('coverage');
gogolon marked this conversation as resolved.
Show resolved Hide resolved

if (!coverageDirectory.existsSync()) {
await coverageDirectory.create();
}
await File(
coverageDirectory.uri.resolve('patrol_lcov.info').toString(),
).writeAsString(report);
}

Future<void> runCodeCoverage({
required String flutterPackageName,
required Directory flutterPackageDirectory,
required TargetPlatform platform,
required Logger logger,
required Set<Glob> ignoreGlobs,
}) async {
final homeDirectory =
Platform.environment['HOME'] ?? Platform.environment['USERPROFILE'];

final logsProcess = await Process.start(
gogolon marked this conversation as resolved.
Show resolved Hide resolved
'flutter',
['logs'],
workingDirectory: homeDirectory,
gogolon marked this conversation as resolved.
Show resolved Hide resolved
);
final vmRegex = RegExp('listening on (http.+)');

final hitMap = <String, HitMap>{};
int? totalTestCount;
var count = 0;

logsProcess.stdout.transform(utf8.decoder).listen(
(line) async {
final vmLink = vmRegex.firstMatch(line)?.group(1);

if (vmLink == null) {
return;
}

final port = RegExp(':([0-9]+)/').firstMatch(vmLink)!.group(1)!;
final auth = RegExp(':$port/(.+)').firstMatch(vmLink)!.group(1);

final String? hostPort;

switch (platform) {
case TargetPlatform.android:
await _forwardAdbPort('61011', port);
piotruela marked this conversation as resolved.
Show resolved Hide resolved

// It is necessary to grab the port from adb forward --list because
// if debugger was attached, the port might be different from the one
// we set
final forwardList = await Process.run('adb', ['forward', '--list']);
final output = forwardList.stdout as String;
hostPort =
RegExp('tcp:([0-9]+) tcp:$port').firstMatch(output)?.group(1);
case TargetPlatform.iOS || TargetPlatform.macOS:
hostPort = port;
default:
hostPort = null;
}

if (hostPort == null) {
logger.err('Failed to obtain Dart VM uri.');
return;
}

final serviceUri = Uri.parse('http://127.0.0.1:$hostPort/$auth');
final serviceClient = await vmServiceConnectUri(
_createWebSocketUri(serviceUri).toString(),
);
await serviceClient.setFlag('pause_isolates_on_exit', 'true');
await serviceClient.streamListen(EventStreams.kIsolate);
serviceClient.onIsolateEvent.listen(
(event) async {
if (event.kind == EventKind.kIsolateRunnable) {
final isolateCoverage = await collect(
serviceUri,
true,
false,
false,
{flutterPackageName},
isolateIds: {event.isolate!.id!},
);
hitMap.merge(
await HitMap.parseJson(
isolateCoverage['coverage'] as List<Map<String, dynamic>>,
),
);
}
},
);

await serviceClient.streamListen('Extension');
serviceClient.onExtensionEvent.listen(
(event) async {
if (event.extensionKind == 'testCount' && totalTestCount == null) {
totalTestCount = event.extensionData!.data['testCount'] as int;
}

if (event.extensionKind == 'waitForCoverageCollection') {
hitMap.merge(
await _collectCoverage(
serviceClient,
serviceUri,
flutterPackageName,
event.extensionData!.data['mainIsolateId'] as String,
),
);
await serviceClient.dispose();
gogolon marked this conversation as resolved.
Show resolved Hide resolved

logger.info('Collected ${++count} / $totalTestCount coverages');

if (count == totalTestCount) {
logsProcess.kill();

logger.info('All coverage gathered, saving');
final report = hitMap.formatLcov(
await Resolver.create(
packagePath: flutterPackageDirectory.path,
),
ignoreGlobs: ignoreGlobs,
);

await _saveCoverage(report);
}
}
},
);
},
);
}
Loading
Loading