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 8 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
18 changes: 18 additions & 0 deletions docs/cli-commands/test.mdx
Original file line number Diff line number Diff line change
Expand Up @@ -110,6 +110,24 @@ patrol test --exclude-tags android
patrol test --exclude-tags='(android||ios)'
```

#### Coverage

To collect coverage from patrol tests, use `--coverage`.

```
patrol test --coverage
```

The LCOV report will be saved to `/coverage/patrol_lcov.info`.

Additionally, you can exclude certain files from the report using glob patterns and `--coverage-ignore` option. For instance,

```
patrol test --coverage --coverage-ignore="**/*.g.dart"
```

excludes all files ending with `.g.dart`.

### Under the hood

`patrol test` basically calls `patrol build` and then runs the built app
Expand Down
18 changes: 9 additions & 9 deletions docs/compatibility-table.mdx
Original file line number Diff line number Diff line change
Expand Up @@ -11,14 +11,14 @@ is by always using the latest version. However,
if for some reason that isn't possible, you can refer to
the table below to assess which version you should use.

| patrol | patrol_cli |
| patrol_cli | patrol |
| -------------- | -------------- |
| 3.11.0 - | 3.2.0 -
| 3.10.0 | 3.1.0 - 3.1.1 |
| 3.6.0 - 3.10.0 | 2.6.5 - 3.0.1 |
| 3.4.0 - 3.5.2 | 2.6.0 - 2.6.4 |
| 3.0.0 - 3.3.0 | 2.3.0 - 2.5.0 |
| 2.3.0 - 2.3.2 | 2.2.0 - 2.2.2 |
| 2.0.1 - 2.2.5 | 2.0.1 - 2.1.5 |
| 3.2.0 - | 3.11.0 - |
| 3.1.0 - 3.1.1 | 3.10.0 |
| 2.6.5 - 3.0.1 | 3.6.0 - 3.10.0 |
| 2.6.0 - 2.6.4 | 3.4.0 - 3.5.2 |
| 2.3.0 - 2.5.0 | 3.0.0 - 3.3.0 |
| 2.2.0 - 2.2.2 | 2.3.0 - 2.3.2 |
| 2.0.1 - 2.1.5 | 2.0.1 - 2.2.5 |
| 2.0.0 | 2.0.0 |
| 1.0.9 - 1.1.11 | 1.1.4 - 1.1.11 |
| 1.1.4 - 1.1.11 | 1.0.9 - 1.1.11 |
3 changes: 2 additions & 1 deletion packages/patrol/CHANGELOG.md
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
## Unreleased
## 3.11.0

- Add code coverage collection support (#2294)
- No throw error in `selectFineLocation` when it's already selected. (#2302)
- Add Option to select tap Location in enterText and enterTextByIndex (#2312)

gogolon marked this conversation as resolved.
Show resolved Hide resolved
Expand Down
3 changes: 3 additions & 0 deletions packages/patrol_cli/CHANGELOG.md
Original file line number Diff line number Diff line change
@@ -1,3 +1,6 @@
# 3.2.0
- Add code coverage collection support (#2294)
gogolon marked this conversation as resolved.
Show resolved Hide resolved

## 3.1.1

- Fix checking `java` version. (#2301)
Expand Down
2 changes: 1 addition & 1 deletion packages/patrol_cli/lib/src/commands/develop.dart
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@ import 'package:patrol_cli/src/base/exceptions.dart';
import 'package:patrol_cli/src/base/extensions/core.dart';
import 'package:patrol_cli/src/base/logger.dart';
import 'package:patrol_cli/src/commands/dart_define_utils.dart';
import 'package:patrol_cli/src/compatibility_checker.dart';
import 'package:patrol_cli/src/compatibility_checker/compatibility_checker.dart';
import 'package:patrol_cli/src/crossplatform/app_options.dart';
import 'package:patrol_cli/src/crossplatform/flutter_tool.dart';
import 'package:patrol_cli/src/dart_defines_reader.dart';
Expand Down
2 changes: 1 addition & 1 deletion packages/patrol_cli/lib/src/commands/test.dart
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@ 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/commands/dart_define_utils.dart';
import 'package:patrol_cli/src/compatibility_checker.dart';
import 'package:patrol_cli/src/compatibility_checker/compatibility_checker.dart';
import 'package:patrol_cli/src/coverage/coverage_tool.dart';
import 'package:patrol_cli/src/crossplatform/app_options.dart';
import 'package:patrol_cli/src/dart_defines_reader.dart';
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@ import 'package:patrol_cli/src/base/exceptions.dart';
import 'package:patrol_cli/src/base/extensions/completer.dart';
import 'package:patrol_cli/src/base/logger.dart';
import 'package:patrol_cli/src/base/process.dart';
import 'package:patrol_cli/src/compatibility_checker/version_comparator.dart';
import 'package:patrol_cli/src/devices.dart';
import 'package:patrol_cli/src/runner/flutter_command.dart';
import 'package:process/process.dart';
Expand Down Expand Up @@ -81,8 +82,13 @@ class CompatibilityChecker {

final cliVersion = Version.parse(constants.version);
final patrolVersion = Version.parse(packageVersion!);
final versionComparator = VersionComparator(
cliVersionRange: _patrolCliVersionRange,
packageVersionRange: _patrolVersionRange,
);

final isCompatible = cliVersion.isCompatibleWith(patrolVersion);
final isCompatible =
versionComparator.isCompatible(cliVersion, patrolVersion);

if (!isCompatible) {
throwToolExit(
Expand Down Expand Up @@ -162,130 +168,78 @@ Future<void> _checkJavaVersion(
}
}

extension _VersionComparator on Version {
/// Checks if the current Patrol CLI version is compatible with the given Patrol package version.
bool isCompatibleWith(Version patrolVersion) {
final cliVersionRange = toRange(_cliVersionRange);
final versionRange = patrolVersion.toRange(_patrolVersionRange);

if (versionRange == null || cliVersionRange == null) {
return false;
}

if (cliToPatrolMap[cliVersionRange] == versionRange) {
return true;
} else {
return false;
}
}

_VersionRange? toRange(List<_VersionRange> versionRangeList) {
for (final versionRange in versionRangeList) {
if (isInRange(versionRange)) {
return versionRange;
}
}
return null;
}

bool isInRange(_VersionRange versionRange) {
return this >= versionRange.min &&
(hasNoUpperBound(versionRange) || this <= versionRange.max);
}

bool hasNoUpperBound(_VersionRange versionRange) {
return versionRange.max == null;
}
}

final _patrolVersionRange = [
_VersionRange(
VersionRange(
min: Version.parse('1.0.9'),
max: Version.parse('1.1.11'),
),
_VersionRange(
VersionRange(
min: Version.parse('2.0.0'),
max: Version.parse('2.0.0'),
),
_VersionRange(
VersionRange(
min: Version.parse('2.0.1'),
max: Version.parse('2.2.5'),
),
_VersionRange(
VersionRange(
min: Version.parse('2.3.0'),
max: Version.parse('2.3.2'),
),
_VersionRange(
VersionRange(
min: Version.parse('3.0.0'),
max: Version.parse('3.3.0'),
),
_VersionRange(
VersionRange(
min: Version.parse('3.4.0'),
max: Version.parse('3.5.2'),
),
_VersionRange(
VersionRange(
min: Version.parse('3.6.0'),
max: Version.parse('3.10.0'),
),
_VersionRange(
VersionRange(
min: Version.parse('3.10.0'),
max: Version.parse('3.10.0'),
),
_VersionRange(
VersionRange(
min: Version.parse('3.11.0'),
),
];

final _cliVersionRange = [
_VersionRange(
final _patrolCliVersionRange = [
VersionRange(
min: Version.parse('1.1.4'),
max: Version.parse('1.1.11'),
),
_VersionRange(
VersionRange(
min: Version.parse('2.0.0'),
max: Version.parse('2.0.0'),
),
_VersionRange(
VersionRange(
min: Version.parse('2.0.1'),
max: Version.parse('2.1.5'),
),
_VersionRange(
VersionRange(
min: Version.parse('2.2.0'),
max: Version.parse('2.2.2'),
),
_VersionRange(
VersionRange(
min: Version.parse('2.3.0'),
max: Version.parse('2.5.0'),
),
_VersionRange(
VersionRange(
min: Version.parse('2.6.0'),
max: Version.parse('2.6.4'),
),
_VersionRange(
VersionRange(
min: Version.parse('2.6.5'),
max: Version.parse('3.0.1'),
),
_VersionRange(
VersionRange(
min: Version.parse('3.1.0'),
max: Version.parse('3.1.1'),
),
_VersionRange(
VersionRange(
min: Version.parse('3.2.0'),
),
];

final cliToPatrolMap = Map.fromIterables(
_cliVersionRange,
_patrolVersionRange,
);

class _VersionRange {
_VersionRange({
required this.min,
// ignore: unused_element
this.max,
});

final Version min;
final Version? max;
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,60 @@
import 'package:version/version.dart';

class VersionComparator {
VersionComparator({
required List<VersionRange> cliVersionRange,
required List<VersionRange> packageVersionRange,
}) : _cliVersionRange = cliVersionRange,
_cliRangesToPackageRangesMap = Map.fromIterables(
cliVersionRange,
packageVersionRange,
);

final List<VersionRange> _cliVersionRange;
final Map<VersionRange, VersionRange> _cliRangesToPackageRangesMap;

/// Checks if the CLI version is compatible with the given package version.
bool isCompatible(Version cliVersion, Version packageVersion) {
final matchingCliVersionRanges =
_getMatchingRanges(cliVersion, _cliVersionRange);

for (final cliVersionRange in matchingCliVersionRanges) {
final packageVersionRange = _cliRangesToPackageRangesMap[cliVersionRange];
if (packageVersionRange != null &&
isInRange(packageVersion, packageVersionRange)) {
return true;
}
}

return false;
}

List<VersionRange> _getMatchingRanges(
Version version,
List<VersionRange> versionRangeList,
) {
return [
for (final versionRange in versionRangeList)
if (isInRange(version, versionRange)) versionRange,
];
}

bool isInRange(Version version, VersionRange versionRange) {
return version >= versionRange.min &&
(_hasNoUpperBound(versionRange) || version <= versionRange.max);
}

bool _hasNoUpperBound(VersionRange versionRange) {
return versionRange.max == null;
}
}

class VersionRange {
VersionRange({
required this.min,
this.max,
});

final Version min;
final Version? max;
}
Original file line number Diff line number Diff line change
Expand Up @@ -20,7 +20,7 @@ import 'package:patrol_cli/src/commands/devices.dart';
import 'package:patrol_cli/src/commands/doctor.dart';
import 'package:patrol_cli/src/commands/test.dart';
import 'package:patrol_cli/src/commands/update.dart';
import 'package:patrol_cli/src/compatibility_checker.dart';
import 'package:patrol_cli/src/compatibility_checker/compatibility_checker.dart';
import 'package:patrol_cli/src/coverage/coverage_tool.dart';
import 'package:patrol_cli/src/crossplatform/flutter_tool.dart';
import 'package:patrol_cli/src/dart_defines_reader.dart';
Expand Down
Loading
Loading