image_cache_tracing_test.dart 4.08 KB
Newer Older
Dan Field's avatar
Dan Field committed
1 2 3 4 5 6 7
// Copyright 2014 The Flutter 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:convert';
import 'dart:developer' as developer;
import 'dart:isolate' as isolate;
8 9
import 'dart:typed_data';
import 'dart:ui' as ui;
Dan Field's avatar
Dan Field committed
10 11 12

import 'package:flutter/painting.dart';
import 'package:flutter_test/flutter_test.dart';
13 14
import 'package:vm_service/vm_service.dart';
import 'package:vm_service/vm_service_io.dart';
Dan Field's avatar
Dan Field committed
15 16

void main() {
17
  VmService vmService;
Dan Field's avatar
Dan Field committed
18 19 20 21 22
  String isolateId;
  setUpAll(() async {
    final developer.ServiceProtocolInfo info = await developer.Service.getInfo();

    if (info.serverUri == null) {
Dan Field's avatar
Dan Field committed
23
      fail('This test _must_ be run with --enable-vmservice.');
Dan Field's avatar
Dan Field committed
24
    }
25 26 27 28

    vmService = await vmServiceConnectUri('ws://localhost:${info.serverUri.port}${info.serverUri.path}ws');
    await vmService.setVMTimelineFlags(<String>['Dart']);
    isolateId = developer.Service.getIsolateID(isolate.Isolate.current);
Dan Field's avatar
Dan Field committed
29 30 31 32 33 34 35

    // Initialize the image cache.
    TestWidgetsFlutterBinding.ensureInitialized();
  });

  test('Image cache tracing', () async {
    final TestImageStreamCompleter completer1 = TestImageStreamCompleter();
36
    final TestImageStreamCompleter completer2 = TestImageStreamCompleter();
Dan Field's avatar
Dan Field committed
37 38 39 40 41 42
    PaintingBinding.instance.imageCache.putIfAbsent(
      'Test',
      () => completer1,
    );
    PaintingBinding.instance.imageCache.clear();

43 44 45 46 47 48 49 50
    // ignore: invalid_use_of_protected_member
    completer2.setImage(const ImageInfo(image: TestImage()));
    PaintingBinding.instance.imageCache.putIfAbsent(
      'Test2',
      () => completer2,
    );
    PaintingBinding.instance.imageCache.evict('Test2');

51
    final Timeline timeline = await vmService.getVMTimeline();
Dan Field's avatar
Dan Field committed
52
    _expectTimelineEvents(
53
      timeline.traceEvents,
Dan Field's avatar
Dan Field committed
54 55 56 57 58 59 60 61 62 63 64 65 66
      <Map<String, dynamic>>[
        <String, dynamic>{
          'name': 'ImageCache.putIfAbsent',
          'args': <String, dynamic>{'key': 'Test', 'isolateId': isolateId}
        },
        <String, dynamic>{
          'name': 'listener',
          'args': <String, dynamic>{'parentId': '1', 'isolateId': isolateId}
        },
        <String, dynamic>{
          'name': 'ImageCache.clear',
          'args': <String, dynamic>{
            'pendingImages': 1,
Dan Field's avatar
Dan Field committed
67 68
            'keepAliveImages': 0,
            'liveImages': 1,
Dan Field's avatar
Dan Field committed
69 70 71 72
            'currentSizeInBytes': 0,
            'isolateId': isolateId,
          }
        },
73 74 75 76 77 78 79 80
        <String, dynamic>{
          'name': 'ImageCache.putIfAbsent',
          'args': <String, dynamic>{'key': 'Test2', 'isolateId': isolateId}
        },
        <String, dynamic>{
          'name': 'ImageCache.evict',
          'args': <String, dynamic>{'sizeInBytes': 0, 'isolateId': isolateId}
        },
Dan Field's avatar
Dan Field committed
81 82 83 84 85
      ],
    );
  }, skip: isBrowser); // uses dart:isolate and io
}

86 87
void _expectTimelineEvents(List<TimelineEvent> events, List<Map<String, dynamic>> expected) {
  for (final TimelineEvent event in events) {
Dan Field's avatar
Dan Field committed
88
    for (int index = 0; index < expected.length; index += 1) {
89
      if (expected[index]['name'] == event.json['name']) {
Dan Field's avatar
Dan Field committed
90
        final Map<String, dynamic> expectedArgs = expected[index]['args'] as Map<String, dynamic>;
91
        final Map<String, dynamic> args = event.json['args'] as Map<String, dynamic>;
Dan Field's avatar
Dan Field committed
92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113
        if (_mapsEqual(expectedArgs, args)) {
          expected.removeAt(index);
        }
      }
    }
  }
  if (expected.isNotEmpty) {
    final String encodedEvents = jsonEncode(events);
    fail('Timeline did not contain expected events: $expected\nactual: $encodedEvents');
  }
}

bool _mapsEqual(Map<String, dynamic> expectedArgs, Map<String, dynamic> args) {
  for (final String key in expectedArgs.keys) {
    if (expectedArgs[key] != args[key]) {
      return false;
    }
  }
  return true;
}

class TestImageStreamCompleter extends ImageStreamCompleter {}
114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129

class TestImage implements ui.Image {
  const TestImage({this.height = 0, this.width = 0});
  @override
  final int height;
  @override
  final int width;

  @override
  void dispose() { }

  @override
  Future<ByteData> toByteData({ ui.ImageByteFormat format = ui.ImageByteFormat.rawRgba }) {
    throw UnimplementedError();
  }
}