Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Submit feedback
Sign in
Toggle navigation
F
Front-End
Project
Project
Details
Activity
Releases
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
abdullh.alsoleman
Front-End
Commits
7edcc92b
Unverified
Commit
7edcc92b
authored
Mar 14, 2020
by
Kate Lovett
Committed by
GitHub
Mar 14, 2020
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Refactoring Gold to enable both Luci & Cirrus support (#49815)
parent
77c46276
Changes
4
Hide whitespace changes
Inline
Side-by-side
Showing
4 changed files
with
338 additions
and
149 deletions
+338
-149
action_sheet.dart
packages/flutter/lib/src/cupertino/action_sheet.dart
+2
-1
flutter_goldens.dart
packages/flutter_goldens/lib/flutter_goldens.dart
+103
-70
flutter_goldens_test.dart
packages/flutter_goldens/test/flutter_goldens_test.dart
+117
-30
skia_client.dart
packages/flutter_goldens_client/lib/skia_client.dart
+116
-48
No files found.
packages/flutter/lib/src/cupertino/action_sheet.dart
View file @
7edcc92b
...
@@ -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
...
...
packages/flutter_goldens/lib/flutter_goldens.dart
View file @
7edcc92b
...
@@ -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
(
Flutter
SkiaGold
FileComparator
.
isAvailableForEnvironment
(
platform
))
{
if
(
Flutter
PostSubmit
FileComparator
.
isAvailableForEnvironment
(
platform
))
{
goldenFileComparator
=
await
Flutter
SkiaGold
FileComparator
.
fromDefaultComparator
(
platform
);
goldenFileComparator
=
await
Flutter
PostSubmit
FileComparator
.
fromDefaultComparator
(
platform
);
}
else
if
(
FlutterPreSubmitFileComparator
.
isAvailableForEnvironment
(
platform
))
{
}
else
if
(
FlutterPreSubmitFileComparator
.
isAvailableForEnvironment
(
platform
))
{
goldenFileComparator
=
await
FlutterPreSubmitFileComparator
.
fromDefaultComparator
(
platform
);
goldenFileComparator
=
await
FlutterPreSubmitFileComparator
.
fromDefaultComparator
(
platform
);
}
else
if
(
FlutterSkipping
Golden
FileComparator
.
isAvailableForEnvironment
(
platform
))
{
}
else
if
(
FlutterSkippingFileComparator
.
isAvailableForEnvironment
(
platform
))
{
goldenFileComparator
=
FlutterSkipping
Golden
FileComparator
.
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 [Flutter
SkiaGold
FileComparator] is utilized during post-submit
/// * The [Flutter
PostSubmit
FileComparator] 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 th
e
/// [Flutter Gold](https://flutter-gold.skia.org) and
manually compar
e
///
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
/// [Flutter
SkiaGold
FileComparator] and the [FlutterPreSubmitFileComparator].
/// [Flutter
PostSubmit
FileComparator] 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
Flutter
SkiaGold
FileComparator
extends
FlutterGoldenFileComparator
{
class
Flutter
PostSubmit
FileComparator
extends
FlutterGoldenFileComparator
{
/// Creates a [Flutter
SkiaGold
FileComparator] that will test golden file
/// Creates a [Flutter
PostSubmit
FileComparator] 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.
Flutter
SkiaGold
FileComparator
(
Flutter
PostSubmit
FileComparator
(
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 [Flutter
SkiaGold
FileComparator] that mirrors the relative
/// Creates a new [Flutter
PostSubmit
FileComparator] 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
<
Flutter
SkiaGold
FileComparator
>
fromDefaultComparator
(
static
Future
<
Flutter
PostSubmit
FileComparator
>
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
Flutter
SkiaGold
FileComparator
(
baseDirectory
.
uri
,
goldens
);
return
Flutter
PostSubmit
FileComparator
(
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.
/// * [Flutter
SkiaGold
FileComparator], another
/// * [Flutter
PostSubmit
FileComparator], 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
///
perform
ed as pre-submit tests with Skia Gold.
///
execut
ed 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
FlutterSkipping
Golden
FileComparator
extends
FlutterGoldenFileComparator
{
class
FlutterSkippingFileComparator
extends
FlutterGoldenFileComparator
{
/// Creates a [FlutterSkipping
Golden
FileComparator] 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.
FlutterSkipping
Golden
FileComparator
(
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 [FlutterSkipping
Golden
FileComparator].
/// Describes the reason for using the [FlutterSkippingFileComparator].
///
///
/// Cannot be null.
/// Cannot be null.
final
String
reason
;
final
String
reason
;
/// Creates a new [FlutterSkipping
Golden
FileComparator] 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
FlutterSkipping
Golden
FileComparator
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
FlutterSkipping
Golden
FileComparator
(
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.
/// * [Flutter
SkiaGold
FileComparator], another
/// * [Flutter
PostSubmit
FileComparator], 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.
/// * [FlutterSkipping
Golden
FileComparator], 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
FlutterSkipping
Golden
FileComparator
(
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
FlutterSkipping
Golden
FileComparator
(
return
FlutterSkippingFileComparator
(
baseDirectory
.
uri
,
baseDirectory
.
uri
,
goldens
,
goldens
,
'SocketException occurred, could not reach Gold. '
'SocketException occurred, could not reach Gold. '
...
...
packages/flutter_goldens/test/flutter_goldens_test.dart
View file @
7edcc92b
...
@@ -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'
,
()
{
Flutter
SkiaGold
FileComparator
comparator
;
Flutter
PostSubmit
FileComparator
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
=
Flutter
SkiaGold
FileComparator
(
comparator
=
Flutter
PostSubmit
FileComparator
(
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
=
Flutter
SkiaGold
FileComparator
(
comparator
=
Flutter
PostSubmit
FileComparator
(
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
(
Flutter
SkiaGold
FileComparator
.
isAvailableForEnvironment
(
platform
),
Flutter
PostSubmit
FileComparator
.
isAvailableForEnvironment
(
platform
),
isTrue
,
isTrue
,
);
);
});
});
...
@@ -535,7 +620,7 @@ void main() {
...
@@ -535,7 +620,7 @@ void main() {
operatingSystem:
'macos'
operatingSystem:
'macos'
);
);
expect
(
expect
(
Flutter
SkiaGold
FileComparator
.
isAvailableForEnvironment
(
platform
),
Flutter
PostSubmit
FileComparator
.
isAvailableForEnvironment
(
platform
),
isFalse
,
isFalse
,
);
);
});
});
...
@@ -551,7 +636,7 @@ void main() {
...
@@ -551,7 +636,7 @@ void main() {
operatingSystem:
'macos'
operatingSystem:
'macos'
);
);
expect
(
expect
(
Flutter
SkiaGold
FileComparator
.
isAvailableForEnvironment
(
platform
),
Flutter
PostSubmit
FileComparator
.
isAvailableForEnvironment
(
platform
),
isFalse
,
isFalse
,
);
);
});
});
...
@@ -566,7 +651,7 @@ void main() {
...
@@ -566,7 +651,7 @@ void main() {
operatingSystem:
'macos'
operatingSystem:
'macos'
);
);
expect
(
expect
(
Flutter
SkiaGold
FileComparator
.
isAvailableForEnvironment
(
platform
),
Flutter
PostSubmit
FileComparator
.
isAvailableForEnvironment
(
platform
),
isFalse
,
isFalse
,
);
);
});
});
...
@@ -583,7 +668,7 @@ void main() {
...
@@ -583,7 +668,7 @@ void main() {
operatingSystem:
'macos'
operatingSystem:
'macos'
);
);
expect
(
expect
(
Flutter
SkiaGold
FileComparator
.
isAvailableForEnvironment
(
platform
),
Flutter
PostSubmit
FileComparator
.
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
(
FlutterSkipping
Golden
FileComparator
.
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
(
FlutterSkipping
Golden
FileComparator
.
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
,
FlutterSkipping
Golden
FileComparator
);
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
,
FlutterSkipping
Golden
FileComparator
);
expect
(
comparator
.
runtimeType
,
FlutterSkippingFileComparator
);
});
});
});
});
});
});
...
...
packages/flutter_goldens_client/lib/skia_client.dart
View file @
7edcc92b
...
@@ -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 [Flutter
SkiaGold
FileComparator].
/// comparison being evaluated by the [Flutter
PostSubmit
FileComparator].
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
,
);
);
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment