blob: db05d632bbe032da27d50af5ef5005905df8d6fe [file] [log] [blame]
// Copyright 2019 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 'package:fixnum/fixnum.dart';
import 'package:flutter/services.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:flutter/material.dart';
import 'package:mockito/mockito.dart';
import 'package:cocoon_service/protos.dart' show CommitStatus, Commit, Stage, Task;
import 'package:app_flutter/agent_dashboard_page.dart';
import 'package:app_flutter/state/build.dart';
import 'package:app_flutter/widgets/task_grid.dart';
import 'package:app_flutter/widgets/task_attempt_summary.dart';
import 'package:app_flutter/widgets/task_box.dart';
import 'package:app_flutter/widgets/task_overlay.dart';
import '../utils/fake_build.dart';
import '../utils/mocks.dart';
import '../utils/task_icons.dart';
import '../utils/wrapper.dart';
class TestGrid extends StatelessWidget {
const TestGrid({
this.buildState,
this.task,
});
final BuildState buildState;
final Task task;
@override
Widget build(BuildContext context) {
return Material(
child: TaskGrid(
buildState: buildState ?? FakeBuildState(),
commitStatuses: <CommitStatus>[
CommitStatus()
..commit = (Commit()
..author = 'Fats Domino'
..sha = '24e8c0a2')
..stages.add(Stage()
..name = 'Stage'
..tasks.addAll(<Task>[task])),
],
),
);
}
}
void main() {
testWidgets('TaskOverlay shows on click', (WidgetTester tester) async {
await precacheTaskIcons(tester);
final Task expectedTask = Task()
..attempts = 3
..stageName = 'devicelab'
..name = 'Tasky McTaskFace'
..reservedForAgentId = 'Agenty McAgentFace'
..reason = 'Because I said so'
..isFlaky = false // As opposed to the next test.
..status = 'Failed';
final String expectedTaskInfoString = 'Attempts: ${expectedTask.attempts}\n'
'Run time: 0 minutes\n'
'Queue time: 0 seconds\n'
'Flaky: ${expectedTask.isFlaky}';
await tester.pumpWidget(
MaterialApp(
home: Scaffold(
body: TestGrid(
task: expectedTask,
),
),
),
);
await tester.pump();
expect(find.text(expectedTask.name), findsNothing);
expect(find.text(expectedTaskInfoString), findsNothing);
expect(find.text(expectedTask.reservedForAgentId), findsNothing);
await expectLater(find.byType(MaterialApp), matchesGoldenFile('task_overlay_test.normal_overlay_closed.png'));
await tester.tapAt(const Offset(TaskBox.cellSize * 1.5, TaskBox.cellSize * 1.5));
await tester.pump();
expect(find.text(expectedTask.name), findsOneWidget);
expect(find.text(expectedTaskInfoString), findsOneWidget);
expect(find.text('SHOW ${expectedTask.reservedForAgentId}'), findsOneWidget);
await expectLater(find.byType(MaterialApp), matchesGoldenFile('task_overlay_test.normal_overlay_open.png'));
// Since the overlay positions itself below the middle of the widget,
// it is safe to click the widget to close it again.
await tester.tapAt(const Offset(TaskBox.cellSize * 1.5, TaskBox.cellSize * 1.5));
await tester.pump();
expect(find.text(expectedTask.name), findsNothing);
expect(find.text(expectedTaskInfoString), findsNothing);
expect(find.text(expectedTask.reservedForAgentId), findsNothing);
await expectLater(find.byType(MaterialApp), matchesGoldenFile('task_overlay_test.normal_overlay_closed.png'));
});
testWidgets('TaskOverlay shows when flaky is true', (WidgetTester tester) async {
await precacheTaskIcons(tester);
final Task flakyTask = Task()
..attempts = 3
..stageName = 'devicelab'
..name = 'Tasky McTaskFace'
..reason = 'Because I said so'
..isFlaky = true // This is the point of this test.
..status = 'Failed';
final String flakyTaskInfoString = 'Attempts: ${flakyTask.attempts}\n'
'Run time: 0 minutes\n'
'Queue time: 0 seconds\n'
'Flaky: true';
await tester.pumpWidget(
MaterialApp(
home: Scaffold(
body: TestGrid(
task: flakyTask,
),
),
),
);
expect(find.text(flakyTask.name), findsNothing);
expect(find.text(flakyTaskInfoString), findsNothing);
expect(find.text(flakyTask.reservedForAgentId), findsNothing);
await expectLater(find.byType(MaterialApp), matchesGoldenFile('task_overlay_test.flaky_overlay_closed.png'));
await tester.tapAt(const Offset(TaskBox.cellSize * 1.5, TaskBox.cellSize * 1.5));
await tester.pump();
expect(find.text(flakyTask.name), findsOneWidget);
expect(find.text(flakyTaskInfoString), findsOneWidget);
expect(find.text('SHOW ${flakyTask.reservedForAgentId}'), findsOneWidget);
await expectLater(find.byType(MaterialApp), matchesGoldenFile('task_overlay_test.flaky_overlay_open.png'));
});
testWidgets('TaskOverlay computes durations correctly', (WidgetTester tester) async {
final Task timeTask = Task()
..attempts = 1
..stageName = 'devicelab'
..name = 'Tasky McTaskFace'
..reason = 'Because I said so'
..isFlaky = true
..createTimestamp = Int64.parseInt('0') // created at 0ms
..startTimestamp = Int64.parseInt('10000') // started after 10 seconds
..endTimestamp = Int64.parseInt('490000'); // ended after 8 minutes
final String timeTaskInfoString = 'Attempts: ${timeTask.attempts}\n'
'Run time: 8 minutes\n'
'Queue time: 10 seconds\n'
'Flaky: true';
await tester.pumpWidget(
MaterialApp(
home: Scaffold(
body: TestGrid(
task: timeTask,
),
),
),
);
expect(find.text(timeTaskInfoString), findsNothing);
// open the overlay to show the task summary
await tester.tapAt(const Offset(TaskBox.cellSize * 1.5, TaskBox.cellSize * 1.5));
await tester.pump();
expect(find.text(timeTaskInfoString), findsOneWidget);
});
testWidgets('TaskOverlay devicelab agent button redirects to agent page', (WidgetTester tester) async {
final Task expectedTask = Task()
..attempts = 3
..stageName = 'devicelab'
..name = 'Tasky McTaskFace'
..reservedForAgentId = 'Agenty McAgentFace'
..reason = 'Because I said so'
..isFlaky = false
..status = 'Succeeded';
await tester.pumpWidget(
FakeInserter(
child: MaterialApp(
home: Scaffold(
body: TestGrid(
task: expectedTask,
),
),
routes: <String, WidgetBuilder>{
AgentDashboardPage.routeName: (BuildContext context) => const AgentDashboardPage(),
},
),
),
);
// The AppBar title for the agent page
expect(find.text('Infra Agents'), findsNothing);
expect(find.text('SHOW ${expectedTask.reservedForAgentId}'), findsNothing);
expect(find.text(expectedTask.reservedForAgentId), findsNothing);
await tester.tapAt(const Offset(TaskBox.cellSize * 1.5, TaskBox.cellSize * 1.5));
await tester.pump();
expect(find.text('Infra Agents'), findsNothing);
expect(find.text('SHOW ${expectedTask.reservedForAgentId}'), findsOneWidget);
expect(find.text(expectedTask.reservedForAgentId), findsNothing);
await tester.tap(find.text('SHOW ${expectedTask.reservedForAgentId}'));
await tester.pump();
await tester.pump(const Duration(seconds: 1));
expect(find.text('Infra Agents'), findsOneWidget);
expect(find.text('SHOW ${expectedTask.reservedForAgentId}'), findsNothing);
// We check that the agent is filtered correctly, which tests if
// the route argument was parsed correctly, by looking for the
// text field that contains the search pattern. (The actual agent
// isn't listed because we don't set up the test data with any
// agents.)
expect(find.widgetWithText(TextField, expectedTask.reservedForAgentId), findsOneWidget);
});
testWidgets('TaskOverlay shows the right message for nondevicelab tasks', (WidgetTester tester) async {
await precacheTaskIcons(tester);
const String expectedTaskInfoString = 'Task was run outside of devicelab';
await tester.pumpWidget(
MaterialApp(
home: Scaffold(
body: TestGrid(
task: Task()
..stageName = 'cirrus'
..status = 'Succeeded',
),
),
),
);
expect(find.text(expectedTaskInfoString), findsNothing);
await expectLater(find.byType(MaterialApp), matchesGoldenFile('task_overlay_test.nondevicelab_closed.png'));
await tester.tapAt(const Offset(TaskBox.cellSize * 1.5, TaskBox.cellSize * 1.5));
await tester.pump();
expect(find.text(expectedTaskInfoString), findsOneWidget);
await expectLater(find.byType(MaterialApp), matchesGoldenFile('task_overlay_test.nondevicelab_open.png'));
});
testWidgets('TaskOverlay shows TaskAttemptSummary for devicelab tasks', (WidgetTester tester) async {
await tester.pumpWidget(
MaterialApp(
home: Scaffold(
body: TestGrid(
task: Task()
..stageName = 'devicelab'
..status = 'Succeeded'
..attempts = 1,
),
),
),
);
expect(find.byType(TaskAttemptSummary), findsNothing);
await tester.tapAt(const Offset(TaskBox.cellSize * 1.5, TaskBox.cellSize * 1.5));
await tester.pump();
expect(find.byType(TaskAttemptSummary), findsOneWidget);
});
testWidgets('TaskOverlay does not show TaskAttemptSummary for tasks outside of devicelab',
(WidgetTester tester) async {
await tester.pumpWidget(
MaterialApp(
home: Scaffold(
body: TestGrid(
task: Task()
..stageName = 'cirrus'
..status = 'Succeeded'
..attempts = 1,
),
),
),
);
expect(find.byType(TaskAttemptSummary), findsNothing);
await tester.tapAt(const Offset(TaskBox.cellSize * 1.5, TaskBox.cellSize * 1.5));
await tester.pump();
expect(find.byType(TaskAttemptSummary), findsNothing);
});
testWidgets('TaskOverlay: successful rerun shows success snackbar message', (WidgetTester tester) async {
final Task expectedTask = Task()
..attempts = 3
..stageName = 'devicelab'
..name = 'Tasky McTaskFace'
..reservedForAgentId = 'Agenty McAgentFace'
..reason = 'Because I said so'
..isFlaky = false;
await tester.pumpWidget(
MaterialApp(
home: Scaffold(
body: TestGrid(
buildState: FakeBuildState(rerunTaskResult: true),
task: expectedTask,
),
),
),
);
// Open the overlay
await tester.tapAt(const Offset(TaskBox.cellSize * 1.5, TaskBox.cellSize * 1.5));
await tester.pump();
expect(find.text(TaskOverlayContents.rerunErrorMessage), findsNothing);
expect(find.text(TaskOverlayContents.rerunSuccessMessage), findsNothing);
// Click the rerun task button
await tester.tap(find.text('RERUN'));
await tester.pump();
await tester.pump(const Duration(milliseconds: 750)); // open animation
expect(find.text(TaskOverlayContents.rerunErrorMessage), findsNothing);
expect(find.text(TaskOverlayContents.rerunSuccessMessage), findsOneWidget);
// Snackbar message should go away after its duration
await tester.pump(TaskOverlayContents.rerunSnackBarDuration);
await tester.pump(const Duration(milliseconds: 1500)); // close animation
expect(find.text(TaskOverlayContents.rerunErrorMessage), findsNothing);
expect(find.text(TaskOverlayContents.rerunSuccessMessage), findsNothing);
});
testWidgets('failed rerun shows error snackbar message', (WidgetTester tester) async {
final Task expectedTask = Task()
..attempts = 3
..stageName = 'devicelab'
..name = 'Tasky McTaskFace'
..reservedForAgentId = 'Agenty McAgentFace'
..reason = 'Because I said so'
..isFlaky = false
..status = 'New';
await tester.pumpWidget(
MaterialApp(
home: Scaffold(
body: TestGrid(
buildState: FakeBuildState(rerunTaskResult: false),
task: expectedTask,
),
),
),
);
// Open the overlay
await tester.tapAt(const Offset(TaskBox.cellSize * 1.5, TaskBox.cellSize * 1.5));
await tester.pump();
expect(find.text(TaskOverlayContents.rerunErrorMessage), findsNothing);
expect(find.text(TaskOverlayContents.rerunSuccessMessage), findsNothing);
// Click the rerun task button
await tester.tap(find.text('RERUN'));
await tester.pump();
await tester.pump(const Duration(milliseconds: 750)); // open animation
expect(find.text(TaskOverlayContents.rerunErrorMessage), findsOneWidget);
expect(find.text(TaskOverlayContents.rerunSuccessMessage), findsNothing);
// Snackbar message should go away after its duration
await tester.pump(TaskOverlayContents.rerunSnackBarDuration);
await tester.pump(const Duration(milliseconds: 1500)); // close animation
expect(find.text(TaskOverlayContents.rerunErrorMessage), findsNothing);
expect(find.text(TaskOverlayContents.rerunSuccessMessage), findsNothing);
});
testWidgets('log button opens log url for public log', (WidgetTester tester) async {
const MethodChannel channel = MethodChannel('plugins.flutter.io/url_launcher');
final List<MethodCall> log = <MethodCall>[];
channel.setMockMethodCallHandler((MethodCall methodCall) async {
log.add(methodCall);
});
final Task publicTask = Task()..stageName = 'cirrus';
await tester.pumpWidget(
MaterialApp(
home: Scaffold(
body: TestGrid(
task: publicTask,
),
),
),
);
// Open the overlay
await tester.tapAt(const Offset(TaskBox.cellSize * 1.5, TaskBox.cellSize * 1.5));
await tester.pump();
// View log
await tester.tap(find.text('DOWNLOAD ALL LOGS'));
await tester.pump();
expect(
log,
<Matcher>[
isMethodCall('launch', arguments: <String, Object>{
'url': 'https://cirrus-ci.com/build/flutter/flutter/24e8c0a2',
'useSafariVC': true,
'useWebView': false,
'enableJavaScript': false,
'enableDomStorage': false,
'universalLinksOnly': false,
'headers': <String, String>{}
})
],
);
});
testWidgets('log button calls build state to download devicelab log', (WidgetTester tester) async {
final Task expectedTask = Task()
..attempts = 3
..stageName = 'devicelab'
..name = 'Tasky McTaskFace'
..reservedForAgentId = 'Agenty McAgentFace'
..reason = 'Because I said so'
..isFlaky = false
..status = 'New';
final MockBuildState buildState = MockBuildState();
when(buildState.moreStatusesExist).thenReturn(true);
when(buildState.downloadLog(any, any)).thenAnswer((_) async => true);
await tester.pumpWidget(
MaterialApp(
home: Scaffold(
body: TestGrid(
buildState: buildState,
task: expectedTask,
),
),
),
);
// Open the overlay
await tester.tapAt(const Offset(TaskBox.cellSize * 1.5, TaskBox.cellSize * 1.5));
await tester.pump();
verifyNever(buildState.downloadLog(any, any));
// Click log button
await tester.tap(find.text('DOWNLOAD ALL LOGS'));
await tester.pump();
verify(buildState.downloadLog(any, any)).called(1);
});
testWidgets('failing to download devicelab log shows error snackbar', (WidgetTester tester) async {
final Task expectedTask = Task()
..attempts = 3
..stageName = 'devicelab'
..name = 'Tasky McTaskFace'
..reservedForAgentId = 'Agenty McAgentFace'
..reason = 'Because I said so'
..isFlaky = false;
final MockBuildState buildState = MockBuildState();
when(buildState.moreStatusesExist).thenReturn(true);
when(buildState.downloadLog(any, any)).thenAnswer((_) async => false);
await tester.pumpWidget(
MaterialApp(
home: Scaffold(
body: TestGrid(
buildState: buildState,
task: expectedTask,
),
),
),
);
// Open the overlay
await tester.tapAt(const Offset(TaskBox.cellSize * 1.5, TaskBox.cellSize * 1.5));
await tester.pump();
// Click log button
await tester.tap(find.text('DOWNLOAD ALL LOGS'));
await tester.pump();
// expect error snackbar to be shown
await tester.pump(const Duration(milliseconds: 750)); // 750ms open animation
expect(find.text(TaskOverlayContents.downloadLogErrorMessage), findsOneWidget);
// Snackbar message should go away after its duration
await tester.pumpAndSettle(TaskOverlayContents.downloadLogSnackBarDuration); // wait the duration
await tester.pump(); // schedule animation
await tester.pump(const Duration(milliseconds: 1500)); // close animation
expect(find.text(TaskOverlayContents.downloadLogErrorMessage), findsNothing);
});
}