mirror of
https://github.com/flutter/flutter.git
synced 2025-06-03 00:51:18 +00:00
232 lines
8.0 KiB
Dart
232 lines
8.0 KiB
Dart
// Copyright 2016 The Chromium Authors. All rights reserved.
|
|
// Use of this source code is governed by a BSD-style license that can be
|
|
// found in the LICENSE file.
|
|
|
|
import 'dart:async';
|
|
import 'dart:convert' show JsonEncoder, JsonDecoder;
|
|
|
|
import 'package:file/file.dart';
|
|
import 'package:file/local.dart';
|
|
import 'package:flutter_driver/flutter_driver.dart';
|
|
import 'package:path/path.dart' as path;
|
|
import 'package:test/test.dart' hide TypeMatcher, isInstanceOf;
|
|
|
|
const FileSystem _fs = LocalFileSystem();
|
|
|
|
// Demos for which timeline data will be collected using
|
|
// FlutterDriver.traceAction().
|
|
//
|
|
// Warning: The number of tests executed with timeline collection enabled
|
|
// significantly impacts heap size of the running app. When run with
|
|
// --trace-startup, as we do in this test, the VM stores trace events in an
|
|
// endless buffer instead of a ring buffer.
|
|
//
|
|
// These names must match GalleryItem titles from kAllGalleryDemos
|
|
// in examples/flutter_gallery/lib/gallery/demos.dart
|
|
const List<String> kProfiledDemos = <String>[
|
|
'Shrine@Studies',
|
|
'Contact profile@Studies',
|
|
'Animation@Studies',
|
|
'Bottom navigation@Material',
|
|
'Buttons@Material',
|
|
'Cards@Material',
|
|
'Chips@Material',
|
|
'Dialogs@Material',
|
|
'Pickers@Material',
|
|
];
|
|
|
|
// There are 3 places where the Gallery demos are traversed.
|
|
// 1- In widget tests such as examples/flutter_gallery/test/smoke_test.dart
|
|
// 2- In driver tests such as examples/flutter_gallery/test_driver/transitions_perf_test.dart
|
|
// 3- In on-device instrumentation tests such as examples/flutter_gallery/test/live_smoketest.dart
|
|
//
|
|
// If you change navigation behavior in the Gallery or in the framework, make
|
|
// sure all 3 are covered.
|
|
|
|
// Demos that will be backed out of within FlutterDriver.runUnsynchronized();
|
|
//
|
|
// These names must match GalleryItem titles from kAllGalleryDemos
|
|
// in examples/flutter_gallery/lib/gallery/demos.dart
|
|
const List<String> kUnsynchronizedDemos = <String>[
|
|
'Progress indicators@Material',
|
|
'Activity Indicator@Cupertino',
|
|
'Video@Media',
|
|
];
|
|
|
|
const List<String> kSkippedDemos = <String>[];
|
|
|
|
// All of the gallery demos, identified as "title@category".
|
|
//
|
|
// These names are reported by the test app, see _handleMessages()
|
|
// in transitions_perf.dart.
|
|
List<String> _allDemos = <String>[];
|
|
|
|
/// Extracts event data from [events] recorded by timeline, validates it, turns
|
|
/// it into a histogram, and saves to a JSON file.
|
|
Future<void> saveDurationsHistogram(List<Map<String, dynamic>> events, String outputPath) async {
|
|
final Map<String, List<int>> durations = <String, List<int>>{};
|
|
Map<String, dynamic> startEvent;
|
|
|
|
// Save the duration of the first frame after each 'Start Transition' event.
|
|
for (Map<String, dynamic> event in events) {
|
|
final String eventName = event['name'];
|
|
if (eventName == 'Start Transition') {
|
|
assert(startEvent == null);
|
|
startEvent = event;
|
|
} else if (startEvent != null && eventName == 'Frame') {
|
|
final String routeName = startEvent['args']['to'];
|
|
durations[routeName] ??= <int>[];
|
|
durations[routeName].add(event['dur']);
|
|
startEvent = null;
|
|
}
|
|
}
|
|
|
|
// Verify that the durations data is valid.
|
|
if (durations.keys.isEmpty)
|
|
throw 'no "Start Transition" timeline events found';
|
|
final Map<String, int> unexpectedValueCounts = <String, int>{};
|
|
durations.forEach((String routeName, List<int> values) {
|
|
if (values.length != 2) {
|
|
unexpectedValueCounts[routeName] = values.length;
|
|
}
|
|
});
|
|
|
|
if (unexpectedValueCounts.isNotEmpty) {
|
|
final StringBuffer error = StringBuffer('Some routes recorded wrong number of values (expected 2 values/route):\n\n');
|
|
unexpectedValueCounts.forEach((String routeName, int count) {
|
|
error.writeln(' - $routeName recorded $count values.');
|
|
});
|
|
error.writeln('\nFull event sequence:');
|
|
final Iterator<Map<String, dynamic>> eventIter = events.iterator;
|
|
String lastEventName = '';
|
|
String lastRouteName = '';
|
|
while (eventIter.moveNext()) {
|
|
final String eventName = eventIter.current['name'];
|
|
|
|
if (!<String>['Start Transition', 'Frame'].contains(eventName))
|
|
continue;
|
|
|
|
final String routeName = eventName == 'Start Transition'
|
|
? eventIter.current['args']['to']
|
|
: '';
|
|
|
|
if (eventName == lastEventName && routeName == lastRouteName) {
|
|
error.write('.');
|
|
} else {
|
|
error.write('\n - $eventName $routeName .');
|
|
}
|
|
|
|
lastEventName = eventName;
|
|
lastRouteName = routeName;
|
|
}
|
|
throw error;
|
|
}
|
|
|
|
// Save the durations Map to a file.
|
|
final File file = await _fs.file(outputPath).create(recursive: true);
|
|
await file.writeAsString(const JsonEncoder.withIndent(' ').convert(durations));
|
|
}
|
|
|
|
/// Scrolls each demo menu item into view, launches it, then returns to the
|
|
/// home screen twice.
|
|
Future<void> runDemos(List<String> demos, FlutterDriver driver) async {
|
|
final SerializableFinder demoList = find.byValueKey('GalleryDemoList');
|
|
String currentDemoCategory;
|
|
|
|
for (String demo in demos) {
|
|
if (kSkippedDemos.contains(demo))
|
|
continue;
|
|
|
|
final String demoName = demo.substring(0, demo.indexOf('@'));
|
|
final String demoCategory = demo.substring(demo.indexOf('@') + 1);
|
|
print('> $demo');
|
|
|
|
if (currentDemoCategory == null) {
|
|
await driver.tap(find.text(demoCategory));
|
|
} else if (currentDemoCategory != demoCategory) {
|
|
await driver.tap(find.byTooltip('Back'));
|
|
await driver.tap(find.text(demoCategory));
|
|
// Scroll back to the top
|
|
await driver.scroll(demoList, 0.0, 10000.0, const Duration(milliseconds: 100));
|
|
}
|
|
currentDemoCategory = demoCategory;
|
|
|
|
final SerializableFinder demoItem = find.text(demoName);
|
|
await driver.scrollUntilVisible(demoList, demoItem,
|
|
dyScroll: -48.0,
|
|
alignment: 0.5,
|
|
timeout: const Duration(seconds: 30),
|
|
);
|
|
|
|
for (int i = 0; i < 2; i += 1) {
|
|
await driver.tap(demoItem); // Launch the demo
|
|
|
|
if (kUnsynchronizedDemos.contains(demo)) {
|
|
await driver.runUnsynchronized<void>(() async {
|
|
await driver.tap(find.pageBack());
|
|
});
|
|
} else {
|
|
await driver.tap(find.pageBack());
|
|
}
|
|
}
|
|
|
|
print('< Success');
|
|
}
|
|
|
|
// Return to the home screen
|
|
await driver.tap(find.byTooltip('Back'));
|
|
}
|
|
|
|
void main([List<String> args = const <String>[]]) {
|
|
group('flutter gallery transitions', () {
|
|
FlutterDriver driver;
|
|
setUpAll(() async {
|
|
driver = await FlutterDriver.connect();
|
|
|
|
if (args.contains('--with_semantics')) {
|
|
print('Enabeling semantics...');
|
|
await driver.setSemantics(true);
|
|
}
|
|
|
|
// See _handleMessages() in transitions_perf.dart.
|
|
_allDemos = List<String>.from(const JsonDecoder().convert(await driver.requestData('demoNames')));
|
|
if (_allDemos.isEmpty)
|
|
throw 'no demo names found';
|
|
});
|
|
|
|
tearDownAll(() async {
|
|
if (driver != null)
|
|
await driver.close();
|
|
});
|
|
|
|
test('all demos', () async {
|
|
|
|
// Collect timeline data for just a limited set of demos to avoid OOMs.
|
|
final Timeline timeline = await driver.traceAction(
|
|
() async {
|
|
await runDemos(kProfiledDemos, driver);
|
|
},
|
|
streams: const <TimelineStream>[
|
|
TimelineStream.dart,
|
|
TimelineStream.embedder,
|
|
],
|
|
);
|
|
|
|
// Save the duration (in microseconds) of the first timeline Frame event
|
|
// that follows a 'Start Transition' event. The Gallery app adds a
|
|
// 'Start Transition' event when a demo is launched (see GalleryItem).
|
|
final TimelineSummary summary = TimelineSummary.summarize(timeline);
|
|
await summary.writeSummaryToFile('transitions', pretty: true);
|
|
final String histogramPath = path.join(testOutputsDirectory, 'transition_durations.timeline.json');
|
|
await saveDurationsHistogram(
|
|
List<Map<String, dynamic>>.from(timeline.json['traceEvents']),
|
|
histogramPath);
|
|
|
|
// Execute the remaining tests.
|
|
final Set<String> unprofiledDemos = Set<String>.from(_allDemos)..removeAll(kProfiledDemos);
|
|
await runDemos(unprofiledDemos.toList(), driver);
|
|
|
|
}, timeout: const Timeout(Duration(minutes: 5)));
|
|
});
|
|
}
|