Unverified Commit 7edcc92b authored by Kate Lovett's avatar Kate Lovett Committed by GitHub

Refactoring Gold to enable both Luci & Cirrus support (#49815)

parent 77c46276
...@@ -85,7 +85,8 @@ const double _kDividerThickness = 1.0; ...@@ -85,7 +85,8 @@ const double _kDividerThickness = 1.0;
/// sheet title and message text style. /// sheet title and message text style.
/// ///
/// To display action buttons that look like standard iOS action sheet buttons, /// To display action buttons that look like standard iOS action sheet buttons,
/// provide [CupertinoActionSheetAction]s for the [actions] given to this action sheet. /// provide [CupertinoActionSheetAction]s for the [actions] given to this action
/// sheet.
/// ///
/// To include a iOS-style cancel button separate from the other buttons, /// To include a iOS-style cancel button separate from the other buttons,
/// provide an [CupertinoActionSheetAction] for the [cancelButton] given to this /// provide an [CupertinoActionSheetAction] for the [cancelButton] given to this
......
...@@ -28,13 +28,13 @@ const String _kFlutterRootKey = 'FLUTTER_ROOT'; ...@@ -28,13 +28,13 @@ const String _kFlutterRootKey = 'FLUTTER_ROOT';
/// instantiated is based on the current testing environment. /// instantiated is based on the current testing environment.
Future<void> main(FutureOr<void> testMain()) async { Future<void> main(FutureOr<void> testMain()) async {
const Platform platform = LocalPlatform(); const Platform platform = LocalPlatform();
if (FlutterSkiaGoldFileComparator.isAvailableForEnvironment(platform)) { if (FlutterPostSubmitFileComparator.isAvailableForEnvironment(platform)) {
goldenFileComparator = await FlutterSkiaGoldFileComparator.fromDefaultComparator(platform); goldenFileComparator = await FlutterPostSubmitFileComparator.fromDefaultComparator(platform);
} else if (FlutterPreSubmitFileComparator.isAvailableForEnvironment(platform)) { } else if (FlutterPreSubmitFileComparator.isAvailableForEnvironment(platform)) {
goldenFileComparator = await FlutterPreSubmitFileComparator.fromDefaultComparator(platform); goldenFileComparator = await FlutterPreSubmitFileComparator.fromDefaultComparator(platform);
} else if (FlutterSkippingGoldenFileComparator.isAvailableForEnvironment(platform)) { } else if (FlutterSkippingFileComparator.isAvailableForEnvironment(platform)) {
goldenFileComparator = FlutterSkippingGoldenFileComparator.fromDefaultComparator( goldenFileComparator = FlutterSkippingFileComparator.fromDefaultComparator(
'Golden file testing is unavailable on LUCI and some Cirrus shards.' 'Golden file testing is not executed on some Cirrus shards.'
); );
} else { } else {
goldenFileComparator = await FlutterLocalFileComparator.fromDefaultComparator(platform); goldenFileComparator = await FlutterLocalFileComparator.fromDefaultComparator(platform);
...@@ -50,7 +50,7 @@ Future<void> main(FutureOr<void> testMain()) async { ...@@ -50,7 +50,7 @@ Future<void> main(FutureOr<void> testMain()) async {
/// different [FlutterGoldenFileComparator]s, depending on the current testing /// different [FlutterGoldenFileComparator]s, depending on the current testing
/// environment. /// environment.
/// ///
/// * The [FlutterSkiaGoldFileComparator] is utilized during post-submit /// * The [FlutterPostSubmitFileComparator] is utilized during post-submit
/// testing, after a pull request has landed on the master branch. This /// testing, after a pull request has landed on the master branch. This
/// comparator uses the [SkiaGoldClient] and the `goldctl` tool to upload /// comparator uses the [SkiaGoldClient] and the `goldctl` tool to upload
/// tests to the [Flutter Gold dashboard](https://flutter-gold.skia.org). /// tests to the [Flutter Gold dashboard](https://flutter-gold.skia.org).
...@@ -58,28 +58,34 @@ Future<void> main(FutureOr<void> testMain()) async { ...@@ -58,28 +58,34 @@ Future<void> main(FutureOr<void> testMain()) async {
/// repository. /// repository.
/// ///
/// * The [FlutterPreSubmitFileComparator] is utilized in pre-submit testing, /// * The [FlutterPreSubmitFileComparator] is utilized in pre-submit testing,
/// before a pull request can land on the master branch. This comparator /// before a pull request lands on the master branch. When authorized, this
/// uses the [SkiaGoldClient] to request the baseline images kept by the /// comparator uses the [SkiaGoldClient] to execute tryjobs, allowing
/// [Flutter Gold dashboard](https://flutter-gold.skia.org). It then /// contributors to view and check in visual differences before landing the
/// compares the current test image to the baseline images using the /// change.
/// standard [GoldenFileComparator.compareLists] to detect any pixel
/// difference. The [SkiaGoldClient] is also used here to check the active
/// ignores from the dashboard, in order to allow intended changes to pass
/// tests.
/// ///
/// * The [FlutterLocalFileComparator] is used for any other tests run outside /// * When unable to authenticate the `goldctl` tool, this comparator
/// of the above conditions. Similar to the /// uses the [SkiaGoldClient] to request the baseline images kept by the
/// [Flutter Gold dashboard](https://flutter-gold.skia.org). It then
/// compares the current test image to the baseline images using the
/// standard [GoldenFileComparator.compareLists] to detect any pixel
/// difference. The [SkiaGoldClient] is also used in this case to check
/// the active ignores from the dashboard, in order to allow intended
/// changes to pass tests.
///
/// * The [FlutterLocalFileComparator] is used for local development testing.
/// Similar to the unauthorized implementation of the
/// [FlutterPreSubmitFileComparator], this comparator will use the /// [FlutterPreSubmitFileComparator], this comparator will use the
/// [SkiaGoldClient] to request baseline images from /// [SkiaGoldClient] to request baseline images from
/// [Flutter Gold](https://flutter-gold.skia.org) and compares for the /// [Flutter Gold](https://flutter-gold.skia.org) and manually compare
/// current test image. If a difference is detected, this comparator will /// pixels. If a difference is detected, this comparator will
/// generate failure output illustrating the found difference. If a baseline /// generate failure output illustrating the found difference. If a baseline
/// is not found for a given test image, it will consider it a new test and /// is not found for a given test image, it will consider it a new test and
/// output the new image for verification. /// output the new image for verification.
/// The [FlutterSkippingGoldenFileComparator] is utilized to skip tests outside ///
/// of the appropriate environments. Currently, tests executing in post-submit /// The [FlutterSkippingFileComparator] is utilized to skip tests outside
/// on the LUCI build environment are skipped, as post-submit checks are done /// of the appropriate environments described above. Currently, some Cirrus
/// on Cirrus. This comparator is also used when an internet connection is /// test shards do not execute golden file testing, and as such do not require
/// a comparator. This comparator is also used when an internet connection is
/// unavailable. /// unavailable.
abstract class FlutterGoldenFileComparator extends GoldenFileComparator { abstract class FlutterGoldenFileComparator extends GoldenFileComparator {
/// Creates a [FlutterGoldenFileComparator] that will resolve golden file /// Creates a [FlutterGoldenFileComparator] that will resolve golden file
...@@ -131,7 +137,7 @@ abstract class FlutterGoldenFileComparator extends GoldenFileComparator { ...@@ -131,7 +137,7 @@ abstract class FlutterGoldenFileComparator extends GoldenFileComparator {
/// Calculate the appropriate basedir for the current test context. /// Calculate the appropriate basedir for the current test context.
/// ///
/// The optional [suffix] argument is used by the /// The optional [suffix] argument is used by the
/// [FlutterSkiaGoldFileComparator] and the [FlutterPreSubmitFileComparator]. /// [FlutterPostSubmitFileComparator] and the [FlutterPreSubmitFileComparator].
/// These [FlutterGoldenFileComparators] randomize their base directories to /// These [FlutterGoldenFileComparators] randomize their base directories to
/// maintain thread safety while using the `goldctl` tool. /// maintain thread safety while using the `goldctl` tool.
@protected @protected
...@@ -148,7 +154,7 @@ abstract class FlutterGoldenFileComparator extends GoldenFileComparator { ...@@ -148,7 +154,7 @@ abstract class FlutterGoldenFileComparator extends GoldenFileComparator {
if (!local) { if (!local) {
comparisonRoot = fs.systemTempDirectory.childDirectory( comparisonRoot = fs.systemTempDirectory.childDirectory(
'skia_goldens$suffix' 'skia_goldens_$suffix'
); );
} else { } else {
comparisonRoot = flutterRoot.childDirectory( comparisonRoot = flutterRoot.childDirectory(
...@@ -184,12 +190,11 @@ abstract class FlutterGoldenFileComparator extends GoldenFileComparator { ...@@ -184,12 +190,11 @@ abstract class FlutterGoldenFileComparator extends GoldenFileComparator {
} }
} }
/// A [FlutterGoldenFileComparator] for testing golden images with Skia Gold. /// A [FlutterGoldenFileComparator] for testing golden images with Skia Gold in
/// post-submit.
/// ///
/// For testing across all platforms, the [SkiaGoldClient] is used to upload /// For testing across all platforms, the [SkiaGoldClient] is used to upload
/// images for framework-related golden tests and process results. Currently /// images for framework-related golden tests and process results.
/// these tests are designed to be run post-submit on Cirrus CI, informed by the
/// environment.
/// ///
/// See also: /// See also:
/// ///
...@@ -201,13 +206,13 @@ abstract class FlutterGoldenFileComparator extends GoldenFileComparator { ...@@ -201,13 +206,13 @@ abstract class FlutterGoldenFileComparator extends GoldenFileComparator {
/// * [FlutterLocalFileComparator], another /// * [FlutterLocalFileComparator], another
/// [FlutterGoldenFileComparator] that tests golden images locally on your /// [FlutterGoldenFileComparator] that tests golden images locally on your
/// current machine. /// current machine.
class FlutterSkiaGoldFileComparator extends FlutterGoldenFileComparator { class FlutterPostSubmitFileComparator extends FlutterGoldenFileComparator {
/// Creates a [FlutterSkiaGoldFileComparator] that will test golden file /// Creates a [FlutterPostSubmitFileComparator] that will test golden file
/// images against Skia Gold. /// images against Skia Gold.
/// ///
/// The [fs] and [platform] parameters are useful in tests, where the default /// The [fs] and [platform] parameters are useful in tests, where the default
/// file system and platform can be replaced by mock instances. /// file system and platform can be replaced by mock instances.
FlutterSkiaGoldFileComparator( FlutterPostSubmitFileComparator(
final Uri basedir, final Uri basedir,
final SkiaGoldClient skiaClient, { final SkiaGoldClient skiaClient, {
final FileSystem fs = const LocalFileSystem(), final FileSystem fs = const LocalFileSystem(),
...@@ -219,12 +224,12 @@ class FlutterSkiaGoldFileComparator extends FlutterGoldenFileComparator { ...@@ -219,12 +224,12 @@ class FlutterSkiaGoldFileComparator extends FlutterGoldenFileComparator {
platform: platform, platform: platform,
); );
/// Creates a new [FlutterSkiaGoldFileComparator] that mirrors the relative /// Creates a new [FlutterPostSubmitFileComparator] that mirrors the relative
/// path resolution of the default [goldenFileComparator]. /// path resolution of the default [goldenFileComparator].
/// ///
/// The [goldens] and [defaultComparator] parameters are visible for testing /// The [goldens] and [defaultComparator] parameters are visible for testing
/// purposes only. /// purposes only.
static Future<FlutterSkiaGoldFileComparator> fromDefaultComparator( static Future<FlutterPostSubmitFileComparator> fromDefaultComparator(
final Platform platform, { final Platform platform, {
SkiaGoldClient goldens, SkiaGoldClient goldens,
LocalFileComparator defaultComparator, LocalFileComparator defaultComparator,
...@@ -238,10 +243,15 @@ class FlutterSkiaGoldFileComparator extends FlutterGoldenFileComparator { ...@@ -238,10 +243,15 @@ class FlutterSkiaGoldFileComparator extends FlutterGoldenFileComparator {
); );
baseDirectory.createSync(recursive: true); baseDirectory.createSync(recursive: true);
goldens ??= SkiaGoldClient(baseDirectory); goldens ??= SkiaGoldClient(
baseDirectory,
ci: platform.environment.containsKey('CIRRUS_CI')
? ContinuousIntegrationEnvironment.cirrus
: ContinuousIntegrationEnvironment.luci,
);
await goldens.auth(); await goldens.auth();
await goldens.imgtestInit(); await goldens.imgtestInit();
return FlutterSkiaGoldFileComparator(baseDirectory.uri, goldens); return FlutterPostSubmitFileComparator(baseDirectory.uri, goldens);
} }
@override @override
...@@ -253,32 +263,40 @@ class FlutterSkiaGoldFileComparator extends FlutterGoldenFileComparator { ...@@ -253,32 +263,40 @@ class FlutterSkiaGoldFileComparator extends FlutterGoldenFileComparator {
return skiaClient.imgtestAdd(golden.path, goldenFile); return skiaClient.imgtestAdd(golden.path, goldenFile);
} }
/// Decides based on the current environment whether goldens tests should be /// Decides based on the current environment if goldens tests should be
/// performed against Skia Gold. /// executed through Skia Gold.
static bool isAvailableForEnvironment(Platform platform) { static bool isAvailableForEnvironment(Platform platform) {
final String cirrusPR = platform.environment['CIRRUS_PR'] ?? ''; final String cirrusPR = platform.environment['CIRRUS_PR'] ?? '';
final String cirrusBranch = platform.environment['CIRRUS_BRANCH'] ?? ''; final String cirrusBranch = platform.environment['CIRRUS_BRANCH'] ?? '';
return platform.environment.containsKey('CIRRUS_CI') final bool cirrusPostSubmit = platform.environment.containsKey('CIRRUS_CI')
&& cirrusPR.isEmpty && cirrusPR.isEmpty
&& cirrusBranch == 'master' && cirrusBranch == 'master'
&& platform.environment.containsKey('GOLD_SERVICE_ACCOUNT'); && platform.environment.containsKey('GOLD_SERVICE_ACCOUNT');
final bool luciPostSubmit = platform.environment.containsKey('SWARMING_TASK_ID')
// Luci tryjob environments contain this value to inform the [FlutterPreSubmitComparator].
&& !platform.environment.containsKey('GOLD_TRYJOB');
return cirrusPostSubmit || luciPostSubmit;
} }
} }
/// A [FlutterGoldenFileComparator] for testing golden images before changes are /// A [FlutterGoldenFileComparator] for testing golden images before changes are
/// merged into the master branch. /// merged into the master branch.
/// ///
/// This comparator utilizes the [SkiaGoldClient] to request baseline images for /// When authorized (on luci and most cirrus testing conditions), the comparator
/// the given device under test for comparison. This comparator is only /// executes tryjobs using the [SkiaGoldClient].
/// initialized during pre-submit testing on Cirrus CI. ///
/// When unauthorized, this comparator utilizes the [SkiaGoldClient] to request
/// baseline images for the given device under test for manual comparison.
/// ///
/// See also: /// See also:
/// ///
/// * [GoldenFileComparator], the abstract class that /// * [GoldenFileComparator], the abstract class that
/// [FlutterGoldenFileComparator] implements. /// [FlutterGoldenFileComparator] implements.
/// * [FlutterSkiaGoldFileComparator], another /// * [FlutterPostSubmitFileComparator], another
/// [FlutterGoldenFileComparator] that uploads tests to the Skia Gold /// [FlutterGoldenFileComparator] that uploads tests to the Skia Gold
/// dashboard. /// dashboard in post-submit.
/// * [FlutterLocalFileComparator], another /// * [FlutterLocalFileComparator], another
/// [FlutterGoldenFileComparator] that tests golden images locally on your /// [FlutterGoldenFileComparator] that tests golden images locally on your
/// current machine. /// current machine.
...@@ -322,10 +340,22 @@ class FlutterPreSubmitFileComparator extends FlutterGoldenFileComparator { ...@@ -322,10 +340,22 @@ class FlutterPreSubmitFileComparator extends FlutterGoldenFileComparator {
if (!baseDirectory.existsSync()) if (!baseDirectory.existsSync())
baseDirectory.createSync(recursive: true); baseDirectory.createSync(recursive: true);
goldens ??= SkiaGoldClient(baseDirectory); goldens ??= SkiaGoldClient(
baseDirectory,
ci: platform.environment.containsKey('CIRRUS_CI')
? ContinuousIntegrationEnvironment.cirrus
: ContinuousIntegrationEnvironment.luci,
);
final bool hasWritePermission = !platform.environment['GOLD_SERVICE_ACCOUNT'].startsWith('ENCRYPTED'); bool onCirrusWithPermission = false;
if (hasWritePermission) { if (platform.environment.containsKey('GOLD_SERVICE_ACCOUNT')) {
// Some contributors may not have permission on Cirrus to decrypt the
// service account.
onCirrusWithPermission =
!platform.environment['GOLD_SERVICE_ACCOUNT'].startsWith('ENCRYPTED');
}
final bool onLuci = platform.environment.containsKey('SWARMING_TASK_ID');
if (onCirrusWithPermission || onLuci) {
await goldens.auth(); await goldens.auth();
await goldens.tryjobInit(); await goldens.tryjobInit();
return _AuthorizedFlutterPreSubmitComparator( return _AuthorizedFlutterPreSubmitComparator(
...@@ -356,13 +386,17 @@ class FlutterPreSubmitFileComparator extends FlutterGoldenFileComparator { ...@@ -356,13 +386,17 @@ class FlutterPreSubmitFileComparator extends FlutterGoldenFileComparator {
return false; return false;
} }
/// Decides based on the current environment whether goldens tests should be /// Decides based on the current environment if goldens tests should be
/// performed as pre-submit tests with Skia Gold. /// executed as pre-submit tests with Skia Gold.
static bool isAvailableForEnvironment(Platform platform) { static bool isAvailableForEnvironment(Platform platform) {
final String cirrusPR = platform.environment['CIRRUS_PR'] ?? ''; final String cirrusPR = platform.environment['CIRRUS_PR'] ?? '';
return platform.environment.containsKey('CIRRUS_CI') final bool cirrusPreSubmit = platform.environment.containsKey('CIRRUS_CI')
&& cirrusPR.isNotEmpty && cirrusPR.isNotEmpty
&& platform.environment.containsKey('GOLD_SERVICE_ACCOUNT'); && platform.environment.containsKey('GOLD_SERVICE_ACCOUNT');
final bool luciPreSubmit = platform.environment.containsKey('SWARMING_TASK_ID')
&& platform.environment.containsKey('GOLD_TRYJOB');
return cirrusPreSubmit || luciPreSubmit;
} }
} }
...@@ -442,19 +476,15 @@ class _UnauthorizedFlutterPreSubmitComparator extends FlutterPreSubmitFileCompar ...@@ -442,19 +476,15 @@ class _UnauthorizedFlutterPreSubmitComparator extends FlutterPreSubmitFileCompar
} }
} }
/// A [FlutterGoldenFileComparator] for controlling post-submit testing /// A [FlutterGoldenFileComparator] for testing conditions that do not execute
/// conditions that do not execute golden file tests. /// golden file tests.
/// ///
/// Currently, this comparator is used in post-submit checks on LUCI and with /// Currently, this comparator is used in some Cirrus test shards, and when an
/// some Cirrus shards that do not run framework tests. This comparator is also /// internet connection is not available for contacting Gold.
/// used when an internet connection is not available for contacting Gold.
/// ///
/// See also: /// See also:
/// ///
/// * [FlutterGoldensRepositoryFileComparator], another /// * [FlutterPostSubmitFileComparator], another [FlutterGoldenFileComparator]
/// [FlutterGoldenFileComparator] that tests golden images using the
/// flutter/goldens repository.
/// * [FlutterSkiaGoldFileComparator], another [FlutterGoldenFileComparator]
/// that tests golden images through Skia Gold. /// that tests golden images through Skia Gold.
/// * [FlutterPreSubmitFileComparator], another /// * [FlutterPreSubmitFileComparator], another
/// [FlutterGoldenFileComparator] that tests golden images before changes are /// [FlutterGoldenFileComparator] that tests golden images before changes are
...@@ -462,24 +492,24 @@ class _UnauthorizedFlutterPreSubmitComparator extends FlutterPreSubmitFileCompar ...@@ -462,24 +492,24 @@ class _UnauthorizedFlutterPreSubmitComparator extends FlutterPreSubmitFileCompar
/// * [FlutterLocalFileComparator], another /// * [FlutterLocalFileComparator], another
/// [FlutterGoldenFileComparator] that tests golden images locally on your /// [FlutterGoldenFileComparator] that tests golden images locally on your
/// current machine. /// current machine.
class FlutterSkippingGoldenFileComparator extends FlutterGoldenFileComparator { class FlutterSkippingFileComparator extends FlutterGoldenFileComparator {
/// Creates a [FlutterSkippingGoldenFileComparator] that will skip tests that /// Creates a [FlutterSkippingFileComparator] that will skip tests that
/// are not in the right environment for golden file testing. /// are not in the right environment for golden file testing.
FlutterSkippingGoldenFileComparator( FlutterSkippingFileComparator(
final Uri basedir, final Uri basedir,
final SkiaGoldClient skiaClient, final SkiaGoldClient skiaClient,
this.reason, this.reason,
) : assert(reason != null), ) : assert(reason != null),
super(basedir, skiaClient); super(basedir, skiaClient);
/// Describes the reason for using the [FlutterSkippingGoldenFileComparator]. /// Describes the reason for using the [FlutterSkippingFileComparator].
/// ///
/// Cannot be null. /// Cannot be null.
final String reason; final String reason;
/// Creates a new [FlutterSkippingGoldenFileComparator] that mirrors the /// Creates a new [FlutterSkippingFileComparator] that mirrors the
/// relative path resolution of the default [goldenFileComparator]. /// relative path resolution of the default [goldenFileComparator].
static FlutterSkippingGoldenFileComparator fromDefaultComparator( static FlutterSkippingFileComparator fromDefaultComparator(
String reason, { String reason, {
LocalFileComparator defaultComparator, LocalFileComparator defaultComparator,
}) { }) {
...@@ -487,7 +517,7 @@ class FlutterSkippingGoldenFileComparator extends FlutterGoldenFileComparator { ...@@ -487,7 +517,7 @@ class FlutterSkippingGoldenFileComparator extends FlutterGoldenFileComparator {
const FileSystem fs = LocalFileSystem(); const FileSystem fs = LocalFileSystem();
final Uri basedir = defaultComparator.basedir; final Uri basedir = defaultComparator.basedir;
final SkiaGoldClient skiaClient = SkiaGoldClient(fs.directory(basedir)); final SkiaGoldClient skiaClient = SkiaGoldClient(fs.directory(basedir));
return FlutterSkippingGoldenFileComparator(basedir, skiaClient, reason); return FlutterSkippingFileComparator(basedir, skiaClient, reason);
} }
@override @override
...@@ -501,8 +531,11 @@ class FlutterSkippingGoldenFileComparator extends FlutterGoldenFileComparator { ...@@ -501,8 +531,11 @@ class FlutterSkippingGoldenFileComparator extends FlutterGoldenFileComparator {
@override @override
Future<void> update(Uri golden, Uint8List imageBytes) => null; Future<void> update(Uri golden, Uint8List imageBytes) => null;
/// Decides based on the current environment whether this comparator should be /// Decides, based on the current environment, if this comparator should be
/// used. /// used.
///
/// If we are in a CI environment, luci or Cirrus, but are not using the other
/// comparators, we skip.
static bool isAvailableForEnvironment(Platform platform) { static bool isAvailableForEnvironment(Platform platform) {
return platform.environment.containsKey('SWARMING_TASK_ID') return platform.environment.containsKey('SWARMING_TASK_ID')
|| platform.environment.containsKey('CIRRUS_CI'); || platform.environment.containsKey('CIRRUS_CI');
...@@ -526,13 +559,13 @@ class FlutterSkippingGoldenFileComparator extends FlutterGoldenFileComparator { ...@@ -526,13 +559,13 @@ class FlutterSkippingGoldenFileComparator extends FlutterGoldenFileComparator {
/// ///
/// * [GoldenFileComparator], the abstract class that /// * [GoldenFileComparator], the abstract class that
/// [FlutterGoldenFileComparator] implements. /// [FlutterGoldenFileComparator] implements.
/// * [FlutterSkiaGoldFileComparator], another /// * [FlutterPostSubmitFileComparator], another
/// [FlutterGoldenFileComparator] that uploads tests to the Skia Gold /// [FlutterGoldenFileComparator] that uploads tests to the Skia Gold
/// dashboard. /// dashboard.
/// * [FlutterPreSubmitFileComparator], another /// * [FlutterPreSubmitFileComparator], another
/// [FlutterGoldenFileComparator] that tests golden images before changes are /// [FlutterGoldenFileComparator] that tests golden images before changes are
/// merged into the master branch. /// merged into the master branch.
/// * [FlutterSkippingGoldenFileComparator], another /// * [FlutterSkippingFileComparator], another
/// [FlutterGoldenFileComparator] that controls post-submit testing /// [FlutterGoldenFileComparator] that controls post-submit testing
/// conditions that do not execute golden file tests. /// conditions that do not execute golden file tests.
class FlutterLocalFileComparator extends FlutterGoldenFileComparator with LocalComparisonOutput { class FlutterLocalFileComparator extends FlutterGoldenFileComparator with LocalComparisonOutput {
...@@ -580,14 +613,14 @@ class FlutterLocalFileComparator extends FlutterGoldenFileComparator with LocalC ...@@ -580,14 +613,14 @@ class FlutterLocalFileComparator extends FlutterGoldenFileComparator with LocalC
try { try {
await goldens.getExpectations(); await goldens.getExpectations();
} on io.OSError catch (_) { } on io.OSError catch (_) {
return FlutterSkippingGoldenFileComparator( return FlutterSkippingFileComparator(
baseDirectory.uri, baseDirectory.uri,
goldens, goldens,
'OSError occurred, could not reach Gold. ' 'OSError occurred, could not reach Gold. '
'Switching to FlutterSkippingGoldenFileComparator.', 'Switching to FlutterSkippingGoldenFileComparator.',
); );
} on io.SocketException catch (_) { } on io.SocketException catch (_) {
return FlutterSkippingGoldenFileComparator( return FlutterSkippingFileComparator(
baseDirectory.uri, baseDirectory.uri,
goldens, goldens,
'SocketException occurred, could not reach Gold. ' 'SocketException occurred, could not reach Gold. '
......
...@@ -108,6 +108,7 @@ void main() { ...@@ -108,6 +108,7 @@ void main() {
process: process, process: process,
platform: platform, platform: platform,
httpClient: mockHttpClient, httpClient: mockHttpClient,
ci: ContinuousIntegrationEnvironment.cirrus,
); );
when(process.run(any)) when(process.run(any))
...@@ -165,6 +166,76 @@ void main() { ...@@ -165,6 +166,76 @@ void main() {
); );
}); });
test('correctly inits tryjob for luci', () async {
platform = FakePlatform(
environment: <String, String>{
'FLUTTER_ROOT': _kFlutterRoot,
'GOLDCTL' : 'goldctl',
'SWARMING_TASK_ID' : '4ae997b50dfd4d11',
'LOGDOG_STREAM_PREFIX' : 'buildbucket/cr-buildbucket.appspot.com/8885996262141582672',
'GOLD_TRYJOB' : 'refs/pull/49815/head',
},
operatingSystem: 'macos'
);
skiaClient = SkiaGoldClient(
workDirectory,
fs: fs,
process: process,
platform: platform,
httpClient: mockHttpClient,
ci: ContinuousIntegrationEnvironment.luci,
);
final List<String> ciArguments = skiaClient.getCIArguments();
expect(
ciArguments,
equals(
<String>[
'--changelist', '49815',
'--cis', 'buildbucket',
'--jobid', '8885996262141582672',
],
),
);
});
test('correctly inits tryjob for cirrus', () async {
platform = FakePlatform(
environment: <String, String>{
'FLUTTER_ROOT': _kFlutterRoot,
'GOLDCTL' : 'goldctl',
'CIRRUS_CI' : 'true',
'CIRRUS_TASK_ID' : '8885996262141582672',
'CIRRUS_PR' : '49815',
},
operatingSystem: 'macos'
);
skiaClient = SkiaGoldClient(
workDirectory,
fs: fs,
process: process,
platform: platform,
httpClient: mockHttpClient,
ci: ContinuousIntegrationEnvironment.cirrus,
);
final List<String> ciArguments = skiaClient.getCIArguments();
expect(
ciArguments,
equals(
<String>[
'--changelist', '49815',
'--cis', 'cirrus',
'--jobid', '8885996262141582672',
],
),
);
});
group('Request Handling', () { group('Request Handling', () {
String testName; String testName;
String pullRequestNumber; String pullRequestNumber;
...@@ -456,12 +527,12 @@ void main() { ...@@ -456,12 +527,12 @@ void main() {
}); });
group('FlutterGoldenFileComparator', () { group('FlutterGoldenFileComparator', () {
FlutterSkiaGoldFileComparator comparator; FlutterPostSubmitFileComparator comparator;
setUp(() { setUp(() {
final Directory basedir = fs.directory('flutter/test/library/') final Directory basedir = fs.directory('flutter/test/library/')
..createSync(recursive: true); ..createSync(recursive: true);
comparator = FlutterSkiaGoldFileComparator( comparator = FlutterPostSubmitFileComparator(
basedir.uri, basedir.uri,
MockSkiaGoldClient(), MockSkiaGoldClient(),
fs: fs, fs: fs,
...@@ -497,7 +568,7 @@ void main() { ...@@ -497,7 +568,7 @@ void main() {
setUp(() { setUp(() {
final Directory basedir = fs.directory('flutter/test/library/') final Directory basedir = fs.directory('flutter/test/library/')
..createSync(recursive: true); ..createSync(recursive: true);
comparator = FlutterSkiaGoldFileComparator( comparator = FlutterPostSubmitFileComparator(
basedir.uri, basedir.uri,
mockSkiaClient, mockSkiaClient,
fs: fs, fs: fs,
...@@ -506,7 +577,21 @@ void main() { ...@@ -506,7 +577,21 @@ void main() {
}); });
group('correctly determines testing environment', () { group('correctly determines testing environment', () {
test('returns true', () { test('returns true for Luci', () {
platform = FakePlatform(
environment: <String, String>{
'FLUTTER_ROOT': _kFlutterRoot,
'SWARMING_TASK_ID' : '12345678990',
},
operatingSystem: 'macos'
);
expect(
FlutterPostSubmitFileComparator.isAvailableForEnvironment(platform),
isTrue,
);
});
test('returns true for Cirrus', () {
platform = FakePlatform( platform = FakePlatform(
environment: <String, String>{ environment: <String, String>{
'FLUTTER_ROOT': _kFlutterRoot, 'FLUTTER_ROOT': _kFlutterRoot,
...@@ -518,7 +603,7 @@ void main() { ...@@ -518,7 +603,7 @@ void main() {
operatingSystem: 'macos' operatingSystem: 'macos'
); );
expect( expect(
FlutterSkiaGoldFileComparator.isAvailableForEnvironment(platform), FlutterPostSubmitFileComparator.isAvailableForEnvironment(platform),
isTrue, isTrue,
); );
}); });
...@@ -535,7 +620,7 @@ void main() { ...@@ -535,7 +620,7 @@ void main() {
operatingSystem: 'macos' operatingSystem: 'macos'
); );
expect( expect(
FlutterSkiaGoldFileComparator.isAvailableForEnvironment(platform), FlutterPostSubmitFileComparator.isAvailableForEnvironment(platform),
isFalse, isFalse,
); );
}); });
...@@ -551,7 +636,7 @@ void main() { ...@@ -551,7 +636,7 @@ void main() {
operatingSystem: 'macos' operatingSystem: 'macos'
); );
expect( expect(
FlutterSkiaGoldFileComparator.isAvailableForEnvironment(platform), FlutterPostSubmitFileComparator.isAvailableForEnvironment(platform),
isFalse, isFalse,
); );
}); });
...@@ -566,7 +651,7 @@ void main() { ...@@ -566,7 +651,7 @@ void main() {
operatingSystem: 'macos' operatingSystem: 'macos'
); );
expect( expect(
FlutterSkiaGoldFileComparator.isAvailableForEnvironment(platform), FlutterPostSubmitFileComparator.isAvailableForEnvironment(platform),
isFalse, isFalse,
); );
}); });
...@@ -583,7 +668,7 @@ void main() { ...@@ -583,7 +668,7 @@ void main() {
operatingSystem: 'macos' operatingSystem: 'macos'
); );
expect( expect(
FlutterSkiaGoldFileComparator.isAvailableForEnvironment(platform), FlutterPostSubmitFileComparator.isAvailableForEnvironment(platform),
isFalse, isFalse,
); );
}); });
...@@ -595,7 +680,7 @@ void main() { ...@@ -595,7 +680,7 @@ void main() {
final MockSkiaGoldClient mockSkiaClient = MockSkiaGoldClient(); final MockSkiaGoldClient mockSkiaClient = MockSkiaGoldClient();
group('correctly determines testing environment', () { group('correctly determines testing environment', () {
test('returns true', () { test('returns true for Cirrus', () {
platform = FakePlatform( platform = FakePlatform(
environment: <String, String>{ environment: <String, String>{
'FLUTTER_ROOT': _kFlutterRoot, 'FLUTTER_ROOT': _kFlutterRoot,
...@@ -611,6 +696,21 @@ void main() { ...@@ -611,6 +696,21 @@ void main() {
); );
}); });
test('returns true for Luci', () {
platform = FakePlatform(
environment: <String, String>{
'FLUTTER_ROOT': _kFlutterRoot,
'SWARMING_TASK_ID' : '12345678990',
'GOLD_TRYJOB' : 'git/ref/12345/head'
},
operatingSystem: 'macos'
);
expect(
FlutterPreSubmitFileComparator.isAvailableForEnvironment(platform),
isTrue,
);
});
test('returns false - no PR', () { test('returns false - no PR', () {
platform = FakePlatform( platform = FakePlatform(
environment: <String, String>{ environment: <String, String>{
...@@ -642,7 +742,7 @@ void main() { ...@@ -642,7 +742,7 @@ void main() {
); );
}); });
test('returns false - not on Cirrus', () { test('returns false - not on Cirrus or Luci', () {
platform = FakePlatform( platform = FakePlatform(
environment: <String, String>{ environment: <String, String>{
'FLUTTER_ROOT': _kFlutterRoot, 'FLUTTER_ROOT': _kFlutterRoot,
...@@ -765,21 +865,7 @@ void main() { ...@@ -765,21 +865,7 @@ void main() {
group('Skipping', () { group('Skipping', () {
group('correctly determines testing environment', () { group('correctly determines testing environment', () {
test('returns true on LUCI', () { test('returns true on Cirrus shards that don\'t run golden tests', () {
platform = FakePlatform(
environment: <String, String>{
'FLUTTER_ROOT': _kFlutterRoot,
'SWARMING_TASK_ID' : '1234567890',
},
operatingSystem: 'macos'
);
expect(
FlutterSkippingGoldenFileComparator.isAvailableForEnvironment(platform),
isTrue,
);
});
test('returns true on Cirrus', () {
platform = FakePlatform( platform = FakePlatform(
environment: <String, String>{ environment: <String, String>{
'FLUTTER_ROOT': _kFlutterRoot, 'FLUTTER_ROOT': _kFlutterRoot,
...@@ -788,10 +874,11 @@ void main() { ...@@ -788,10 +874,11 @@ void main() {
operatingSystem: 'macos' operatingSystem: 'macos'
); );
expect( expect(
FlutterSkippingGoldenFileComparator.isAvailableForEnvironment(platform), FlutterSkippingFileComparator.isAvailableForEnvironment(platform),
isTrue, isTrue,
); );
}); });
test('returns false - no CI', () { test('returns false - no CI', () {
platform = FakePlatform( platform = FakePlatform(
environment: <String, String>{ environment: <String, String>{
...@@ -800,7 +887,7 @@ void main() { ...@@ -800,7 +887,7 @@ void main() {
operatingSystem: 'macos' operatingSystem: 'macos'
); );
expect( expect(
FlutterSkippingGoldenFileComparator.isAvailableForEnvironment( FlutterSkippingFileComparator.isAvailableForEnvironment(
platform), platform),
isFalse, isFalse,
); );
...@@ -891,7 +978,7 @@ void main() { ...@@ -891,7 +978,7 @@ void main() {
goldens: mockSkiaClient, goldens: mockSkiaClient,
baseDirectory: mockDirectory, baseDirectory: mockDirectory,
); );
expect(comparator.runtimeType, FlutterSkippingGoldenFileComparator); expect(comparator.runtimeType, FlutterSkippingFileComparator);
when(mockSkiaClient.getExpectations()) when(mockSkiaClient.getExpectations())
.thenAnswer((_) => throw const SocketException("Can't reach Gold")); .thenAnswer((_) => throw const SocketException("Can't reach Gold"));
...@@ -900,7 +987,7 @@ void main() { ...@@ -900,7 +987,7 @@ void main() {
goldens: mockSkiaClient, goldens: mockSkiaClient,
baseDirectory: mockDirectory, baseDirectory: mockDirectory,
); );
expect(comparator.runtimeType, FlutterSkippingGoldenFileComparator); expect(comparator.runtimeType, FlutterSkippingFileComparator);
}); });
}); });
}); });
......
...@@ -21,6 +21,12 @@ const String _kGoldctlKey = 'GOLDCTL'; ...@@ -21,6 +21,12 @@ const String _kGoldctlKey = 'GOLDCTL';
const String _kServiceAccountKey = 'GOLD_SERVICE_ACCOUNT'; const String _kServiceAccountKey = 'GOLD_SERVICE_ACCOUNT';
const String _kTestBrowserKey = 'FLUTTER_TEST_BROWSER'; const String _kTestBrowserKey = 'FLUTTER_TEST_BROWSER';
/// Enum representing the supported CI environments used by flutter/flutter.
enum ContinuousIntegrationEnvironment {
luci,
cirrus,
}
/// A client for uploading image tests and making baseline requests to the /// A client for uploading image tests and making baseline requests to the
/// Flutter Gold Dashboard. /// Flutter Gold Dashboard.
class SkiaGoldClient { class SkiaGoldClient {
...@@ -29,6 +35,7 @@ class SkiaGoldClient { ...@@ -29,6 +35,7 @@ class SkiaGoldClient {
this.fs = const LocalFileSystem(), this.fs = const LocalFileSystem(),
this.process = const LocalProcessManager(), this.process = const LocalProcessManager(),
this.platform = const LocalPlatform(), this.platform = const LocalPlatform(),
this.ci,
io.HttpClient httpClient, io.HttpClient httpClient,
}) : assert(workDirectory != null), }) : assert(workDirectory != null),
assert(fs != null), assert(fs != null),
...@@ -38,23 +45,26 @@ class SkiaGoldClient { ...@@ -38,23 +45,26 @@ class SkiaGoldClient {
/// The file system to use for storing the local clone of the repository. /// The file system to use for storing the local clone of the repository.
/// ///
/// This is useful in tests, where a local file system (the default) can /// This is useful in tests, where a local file system (the default) can be
/// be replaced by a memory file system. /// replaced by a memory file system.
final FileSystem fs; final FileSystem fs;
/// A wrapper for the [dart:io.Platform] API. /// A wrapper for the [dart:io.Platform] API.
/// ///
/// This is useful in tests, where the system platform (the default) can /// This is useful in tests, where the system platform (the default) can be
/// be replaced by a mock platform instance. /// replaced by a mock platform instance.
final Platform platform; final Platform platform;
/// A controller for launching sub-processes. /// A controller for launching sub-processes.
/// ///
/// This is useful in tests, where the real process manager (the default) /// This is useful in tests, where the real process manager (the default) can
/// can be replaced by a mock process manager that doesn't really create /// be replaced by a mock process manager that doesn't really create
/// sub-processes. /// sub-processes.
final ProcessManager process; final ProcessManager process;
/// What testing environment we may be in, like Cirrus or Luci.
final ContinuousIntegrationEnvironment ci;
/// A client for making Http requests to the Flutter Gold dashboard. /// A client for making Http requests to the Flutter Gold dashboard.
final io.HttpClient httpClient; final io.HttpClient httpClient;
...@@ -69,9 +79,9 @@ class SkiaGoldClient { ...@@ -69,9 +79,9 @@ class SkiaGoldClient {
/// A map of known golden file tests and their associated positive image /// A map of known golden file tests and their associated positive image
/// hashes. /// hashes.
/// ///
/// This is set and used by the [FlutterLocalFileComparator] and /// This is set and used by the [FlutterLocalFileComparator] and the
/// [FlutterPreSubmitFileComparator] to test against golden masters maintained /// [_UnauthorizedFlutterPreSubmitComparator] to test against golden masters
/// in the Flutter Gold dashboard. /// maintained in the Flutter Gold dashboard.
Map<String, List<String>> get expectations => _expectations; Map<String, List<String>> get expectations => _expectations;
Map<String, List<String>> _expectations; Map<String, List<String>> _expectations;
...@@ -94,31 +104,57 @@ class SkiaGoldClient { ...@@ -94,31 +104,57 @@ class SkiaGoldClient {
/// Prepares the local work space for golden file testing and calls the /// Prepares the local work space for golden file testing and calls the
/// goldctl `auth` command. /// goldctl `auth` command.
/// ///
/// This ensures that the goldctl tool is authorized and ready for testing. It /// This ensures that the goldctl tool is authorized and ready for testing.
/// will only be called once for each instance of /// Used by the [FlutterPostSubmitFileComparator] and the
/// [FlutterSkiaGoldFileComparator]. /// [_AuthorizedFlutterPreSubmitComparator].
///
/// Based on the current environment, the goldctl tool may be authorized by
/// a service account provided by Cirrus, or through the context provided by a
/// luci environment.
Future<void> auth() async { Future<void> auth() async {
if (await clientIsAuthorized()) if (await clientIsAuthorized())
return; return;
if (_serviceAccount.isEmpty) { List<String> authArguments;
final StringBuffer buf = StringBuffer() String failureContext;
..writeln('The Gold service account is unavailable.')
..writeln('Without a service account, Gold can not be authorized.') switch (ci) {
..writeln('Please check your user permissions and current comparator.'); case ContinuousIntegrationEnvironment.luci:
throw Exception(buf.toString()); authArguments = <String>[
} 'auth',
'--work-dir', workDirectory
final File authorization = workDirectory.childFile('serviceAccount.json'); .childDirectory('temp')
await authorization.writeAsString(_serviceAccount); .path,
'--luci',
];
failureContext =
'Luci environments authenticate using the file provided '
'by LUCI_CONTEXT. There may be an error with this file or Gold '
'authentication.';
break;
case ContinuousIntegrationEnvironment.cirrus:
if (_serviceAccount.isEmpty) {
final StringBuffer buf = StringBuffer()
..writeln('The Gold service account is unavailable.')..writeln(
'Without a service account, Gold can not be authorized.')..writeln(
'Please check your user permissions and current comparator.');
throw Exception(buf.toString());
}
final List<String> authArguments = <String>[ final File authorization = workDirectory.childFile('serviceAccount.json');
'auth', await authorization.writeAsString(_serviceAccount);
'--service-account', authorization.path, authArguments = <String>[
'--work-dir', workDirectory 'auth',
.childDirectory('temp') '--service-account', authorization.path,
.path, '--work-dir', workDirectory
]; .childDirectory('temp')
.path,
];
failureContext = 'This could be caused by incorrect user permissions on '
'Cirrus, if the debug information below contains ENCRYPTED, the wrong '
'comparator was chosen for the test case.';
break;
}
final io.ProcessResult result = await io.Process.run( final io.ProcessResult result = await io.Process.run(
_goldctl, _goldctl,
...@@ -128,10 +164,7 @@ class SkiaGoldClient { ...@@ -128,10 +164,7 @@ class SkiaGoldClient {
if (result.exitCode != 0) { if (result.exitCode != 0) {
final StringBuffer buf = StringBuffer() final StringBuffer buf = StringBuffer()
..writeln('Skia Gold authorization failed.') ..writeln('Skia Gold authorization failed.')
..writeln('This could be caused by incorrect user permissions, if the ') ..writeln(failureContext)
..writeln('debug information below contains ENCRYPTED, the wrong ')
..writeln('comparator was chosen for the test case.')
..writeln()
..writeln('Debug information for Gold:') ..writeln('Debug information for Gold:')
..writeln('stdout: ${result.stdout}') ..writeln('stdout: ${result.stdout}')
..writeln('stderr: ${result.stderr}'); ..writeln('stderr: ${result.stderr}');
...@@ -145,6 +178,10 @@ class SkiaGoldClient { ...@@ -145,6 +178,10 @@ class SkiaGoldClient {
/// It will only be called once for each instance of an /// It will only be called once for each instance of an
/// [_UnauthorizedFlutterPreSubmitComparator]. /// [_UnauthorizedFlutterPreSubmitComparator].
Future<void> emptyAuth() async { Future<void> emptyAuth() async {
// We only use emptyAuth when the service account cannot be decrypted on
// Cirrus.
assert(ci == ContinuousIntegrationEnvironment.cirrus);
final List<String> authArguments = <String>[ final List<String> authArguments = <String>[
'auth', 'auth',
'--work-dir', workDirectory '--work-dir', workDirectory
...@@ -171,7 +208,8 @@ class SkiaGoldClient { ...@@ -171,7 +208,8 @@ class SkiaGoldClient {
/// Executes the `imgtest init` command in the goldctl tool. /// Executes the `imgtest init` command in the goldctl tool.
/// ///
/// The `imgtest` command collects and uploads test results to the Skia Gold /// The `imgtest` command collects and uploads test results to the Skia Gold
/// backend, the `init` argument initializes the current test. /// backend, the `init` argument initializes the current test. Used by the
/// [FlutterPostSubmitFileComparator].
Future<void> imgtestInit() async { Future<void> imgtestInit() async {
final File keys = workDirectory.childFile('keys.json'); final File keys = workDirectory.childFile('keys.json');
final File failures = workDirectory.childFile('failures.json'); final File failures = workDirectory.childFile('failures.json');
...@@ -227,7 +265,7 @@ class SkiaGoldClient { ...@@ -227,7 +265,7 @@ class SkiaGoldClient {
/// result. /// result.
/// ///
/// The [testName] and [goldenFile] parameters reference the current /// The [testName] and [goldenFile] parameters reference the current
/// comparison being evaluated by the [FlutterSkiaGoldFileComparator]. /// comparison being evaluated by the [FlutterPostSubmitFileComparator].
Future<bool> imgtestAdd(String testName, File goldenFile) async { Future<bool> imgtestAdd(String testName, File goldenFile) async {
assert(testName != null); assert(testName != null);
assert(goldenFile != null); assert(goldenFile != null);
...@@ -260,7 +298,8 @@ class SkiaGoldClient { ...@@ -260,7 +298,8 @@ class SkiaGoldClient {
/// Executes the `imgtest init` command in the goldctl tool for tryjobs. /// Executes the `imgtest init` command in the goldctl tool for tryjobs.
/// ///
/// The `imgtest` command collects and uploads test results to the Skia Gold /// The `imgtest` command collects and uploads test results to the Skia Gold
/// backend, the `init` argument initializes the current tryjob. /// backend, the `init` argument initializes the current tryjob. Used by the
/// [_AuthorizedFlutterPreSubmitComparator].
Future<void> tryjobInit() async { Future<void> tryjobInit() async {
final File keys = workDirectory.childFile('keys.json'); final File keys = workDirectory.childFile('keys.json');
final File failures = workDirectory.childFile('failures.json'); final File failures = workDirectory.childFile('failures.json');
...@@ -268,9 +307,6 @@ class SkiaGoldClient { ...@@ -268,9 +307,6 @@ class SkiaGoldClient {
await keys.writeAsString(_getKeysJSON()); await keys.writeAsString(_getKeysJSON());
await failures.create(); await failures.create();
final String commitHash = await _getCurrentCommit(); final String commitHash = await _getCurrentCommit();
final String pullRequest = platform.environment['CIRRUS_PR'];
final String cirrusTaskID = platform.environment['CIRRUS_TASK_ID'];
final List<String> imgtestInitArguments = <String>[ final List<String> imgtestInitArguments = <String>[
'imgtest', 'init', 'imgtest', 'init',
...@@ -283,12 +319,11 @@ class SkiaGoldClient { ...@@ -283,12 +319,11 @@ class SkiaGoldClient {
'--failure-file', failures.path, '--failure-file', failures.path,
'--passfail', '--passfail',
'--crs', 'github', '--crs', 'github',
'--changelist', pullRequest,
'--cis', 'cirrus',
'--jobid', cirrusTaskID,
'--patchset_id', commitHash, '--patchset_id', commitHash,
]; ];
imgtestInitArguments.addAll(getCIArguments());
if (imgtestInitArguments.contains(null)) { if (imgtestInitArguments.contains(null)) {
final StringBuffer buf = StringBuffer() final StringBuffer buf = StringBuffer()
..writeln('A null argument was provided for Skia Gold tryjob init.') ..writeln('A null argument was provided for Skia Gold tryjob init.')
...@@ -452,11 +487,12 @@ class SkiaGoldClient { ...@@ -452,11 +487,12 @@ class SkiaGoldClient {
/// Returns a boolean value for whether or not the given test and current pull /// Returns a boolean value for whether or not the given test and current pull
/// request are ignored on Flutter Gold. /// request are ignored on Flutter Gold.
/// ///
/// This is only relevant when used by the [FlutterPreSubmitFileComparator] /// This is only relevant when used by the
/// when a golden file test fails. In order to land a change to an existing /// [_UnauthorizedFlutterPreSubmitComparator] when a golden file test fails.
/// golden file, an ignore must be set up in Flutter Gold. This will serve as /// In order to land a change to an existing golden file, an ignore must be
/// a flag to permit the change to land, protect against any unwanted changes, /// set up in Flutter Gold. This will serve as a flag to permit the change to
/// and ensure that changes that have landed are triaged. /// land, protect against any unwanted changes, and ensure that changes that
/// have landed are triaged.
Future<bool> testIsIgnoredForPullRequest(String pullRequest, String testName) async { Future<bool> testIsIgnoredForPullRequest(String pullRequest, String testName) async {
bool ignoreIsActive = false; bool ignoreIsActive = false;
testName = cleanTestName(testName); testName = cleanTestName(testName);
...@@ -574,6 +610,7 @@ class SkiaGoldClient { ...@@ -574,6 +610,7 @@ class SkiaGoldClient {
String _getKeysJSON() { String _getKeysJSON() {
final Map<String, dynamic> keys = <String, dynamic>{ final Map<String, dynamic> keys = <String, dynamic>{
'Platform' : platform.operatingSystem, 'Platform' : platform.operatingSystem,
'CI' : ci.toString().split('.').last,
}; };
if (platform.environment[_kTestBrowserKey] != null) if (platform.environment[_kTestBrowserKey] != null)
keys['Browser'] = platform.environment[_kTestBrowserKey]; keys['Browser'] = platform.environment[_kTestBrowserKey];
...@@ -601,6 +638,34 @@ class SkiaGoldClient { ...@@ -601,6 +638,34 @@ class SkiaGoldClient {
} }
return false; return false;
} }
/// Returns a list of arguments for initializing a tryjob based on the testing
/// environment.
List<String> getCIArguments() {
String pullRequest;
String jobId;
String cis;
switch (ci) {
case ContinuousIntegrationEnvironment.luci:
jobId = platform.environment['LOGDOG_STREAM_PREFIX'].split('/').last;
final List<String> refs = platform.environment['GOLD_TRYJOB'].split('/');
pullRequest = refs[refs.length - 2];
cis = 'buildbucket';
break;
case ContinuousIntegrationEnvironment.cirrus:
pullRequest = platform.environment['CIRRUS_PR'];
jobId = platform.environment['CIRRUS_TASK_ID'];
cis = 'cirrus';
break;
}
return <String>[
'--changelist', pullRequest,
'--cis', cis,
'--jobid', jobId,
];
}
} }
/// Used to make HttpRequests during testing. /// Used to make HttpRequests during testing.
...@@ -623,7 +688,10 @@ class SkiaGoldDigest { ...@@ -623,7 +688,10 @@ class SkiaGoldDigest {
return SkiaGoldDigest( return SkiaGoldDigest(
imageHash: json['digest'] as String, imageHash: json['digest'] as String,
paramSet: Map<String, dynamic>.from(json['paramset'] as Map<String, dynamic> ?? paramSet: Map<String, dynamic>.from(json['paramset'] as Map<String, dynamic> ??
<String, List<String>>{'Platform': <String>[]}), <String, List<String>>{
'Platform': <String>[],
'Browser' : <String>[],
}),
testName: json['test'] as String, testName: json['test'] as String,
status: json['status'] as String, status: json['status'] as String,
); );
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment