gallery.dart 3.06 KB
Newer Older
1 2 3 4 5 6 7
// 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';
import 'dart:io';
8
import 'dart:math' as math;
9 10 11

import '../framework/adb.dart';
import '../framework/framework.dart';
12
import '../framework/ios.dart';
13 14
import '../framework/utils.dart';

15 16
TaskFunction createGalleryTransitionTest({ bool semanticsEnabled: false }) {
  return new GalleryTransitionTest(semanticsEnabled: semanticsEnabled);
17 18 19 20
}

class GalleryTransitionTest {

21 22 23 24
  GalleryTransitionTest({ this.semanticsEnabled: false });

  final bool semanticsEnabled;

25
  Future<TaskResult> call() async {
26
    final Device device = await devices.workingDevice;
27
    await device.unlock();
28 29
    final String deviceId = device.deviceId;
    final Directory galleryDirectory =
30 31
        dir('${flutterDirectory.path}/examples/flutter_gallery');
    await inDirectory(galleryDirectory, () async {
32
      await flutter('packages', options: <String>['get']);
33

34
      if (deviceOperatingSystem == DeviceOperatingSystem.ios)
35
        await prepareProvisioningCertificates(galleryDirectory.path);
36

37 38 39 40
      final String testDriver = semanticsEnabled
          ? 'transitions_perf_with_semantics.dart'
          : 'transitions_perf.dart';

41 42 43 44
      await flutter('drive', options: <String>[
        '--profile',
        '--trace-startup',
        '-t',
45
        'test_driver/$testDriver',
46 47 48 49 50 51 52
        '-d',
        deviceId,
      ]);
    });

    // Route paths contains slashes, which Firebase doesn't accept in keys, so we
    // remove them.
53
    final Map<String, List<int>> original = JSON.decode(file(
54 55
            '${galleryDirectory.path}/build/transition_durations.timeline.json')
        .readAsStringSync());
56 57 58 59
    final Map<String, List<int>> transitions = <String, List<int>>{};
    for (String key in original.keys) {
      transitions[key.replaceAll('/', '')] = original[key];
    }
60

61
    final Map<String, dynamic> summary = JSON.decode(file('${galleryDirectory.path}/build/transitions.timeline_summary.json').readAsStringSync());
62

63
    final Map<String, dynamic> data = <String, dynamic>{
64
      'transitions': transitions,
65
      'missed_transition_count': _countMissedTransitions(transitions),
66 67 68 69
    };
    data.addAll(summary);

    return new TaskResult.success(data, benchmarkScoreKeys: <String>[
70
      'missed_transition_count',
71 72 73
      'average_frame_build_time_millis',
      'worst_frame_build_time_millis',
      'missed_frame_build_budget_count',
74 75 76
      'average_frame_rasterizer_time_millis',
      'worst_frame_rasterizer_time_millis',
      'missed_frame_rasterizer_budget_count',
77
    ]);
78 79
  }
}
80 81 82 83 84

int _countMissedTransitions(Map<String, List<int>> transitions) {
  const int _kTransitionBudget = 100000; // µs
  int count = 0;
  transitions.forEach((String demoName, List<int> durations) {
85
    final int longestDuration = durations.reduce(math.max);
86 87 88 89 90 91 92
    if (longestDuration > _kTransitionBudget) {
      print('$demoName missed transition time budget ($longestDuration µs > $_kTransitionBudget µs)');
      count++;
    }
  });
  return count;
}