flutter-tester

This skill provides comprehensive guidance for writing consistent, reliable, and maintainable tests for Flutter applications. Follow the testing patterns, mocking strategies, and architectural guidelines to ensure tests are isolated, repeatable, and cover both success and error scenarios. This skill works with any Flutter project using common packages like Riverpod, Mockito, and flutter_test.

Safety Notice

This listing is imported from skills.sh public index metadata. Review upstream SKILL.md and repository scripts before running.

Copy this and send it to your AI assistant to learn

Install skill "flutter-tester" with this command: npx skills add rimthan-lab/rimthan-plugins/rimthan-lab-rimthan-plugins-flutter-tester

Flutter Tester

Overview

This skill provides comprehensive guidance for writing consistent, reliable, and maintainable tests for Flutter applications. Follow the testing patterns, mocking strategies, and architectural guidelines to ensure tests are isolated, repeatable, and cover both success and error scenarios. This skill works with any Flutter project using common packages like Riverpod, Mockito, and flutter_test.

When to Use This Skill

Use this skill when:

  • Creating new unit tests for repositories, providers, DAOs, or services

  • Writing widget tests for UI components and views

  • Setting up mocks and test dependencies with Mockito and Riverpod

  • Implementing Given-When-Then test structure

  • Testing state management with Riverpod providers

  • Writing integration tests for multi-layer workflows

  • Debugging or fixing existing tests

  • Ensuring proper test coverage across data, domain, and presentation layers

Core Testing Principles

  1. Clean Architecture Testing

Test each layer in isolation:

  • Data Layer → DAOs, APIs, Repositories

  • Domain Layer → Models (Freezed), Entities

  • Presentation Layer → Providers (Riverpod), Views, Controllers

  1. Given-When-Then Structure

Always structure tests using Given-When-Then pattern:

test('Given valid data, When operation executes, Then returns expected result', () async { // Arrange (Given) when(mockDAO.getData()).thenAnswer((_) async => testData);

// Act (When) final result = await repository.fetchData();

// Assert (Then) expect(result, equals(testData)); verify(mockDAO.getData()).called(1); });

  1. Test Organization
  • Group related tests using group() blocks

  • Use setUp() for common initialization

  • Use tearDown() for cleanup (reset GetIt, dispose resources)

  • Use setUpAll() for one-time expensive setup

Testing Workflow

Step 1: Identify the Layer Under Test

Determine which architectural layer you're testing:

  • Repository tests → Mock DAOs and APIs

  • Provider tests → Mock services and repositories

  • Widget tests → Mock providers and services

  • DAO tests → Use FakeDatabase

Step 2: Set Up Dependencies and Mocks

Generate Mocks with Mockito

@GenerateMocks([ILogger, ICarouselRepository, INotificationDAO]) void main() { // Test code }

Important: Never mock providers directly. Override their dependencies instead.

Register with GetIt

setUp(() { mockLogger = MockILogger(); mockRepository = MockICarouselRepository(); GetIt.I.registerSingleton<ILogger>(mockLogger); GetIt.I.registerSingleton<ICarouselRepository>(mockRepository); });

tearDown(() => GetIt.I.reset());

SharedPreferences Setup

setUpAll(() async { SharedPreferences.setMockInitialValues({'key1': 'value1'}); SharedPrefManager.instance = await SharedPreferences.getInstance(); });

Step 3: Write Tests Following Layer Patterns

Refer to the references/layer_testing_patterns.md file for detailed examples of:

  • Repository testing patterns

  • Provider testing patterns with Riverpod

  • DAO testing patterns with FakeDatabase

  • Widget testing patterns with keys and screen size setup

Step 4: Test Error Scenarios

Always test both success and failure paths:

test('Given service throws exception, When called, Then logs error and returns fallback', () async { // Arrange final exception = Exception('Network error'); when(mockService.fetchData()).thenThrow(exception);

// Act final result = await repository.getData();

// Assert expect(result, isEmpty); // Or appropriate fallback verify(mockLogger.writeExceptionLog('RepositoryName', 'getData', exception, any)).called(1); });

Widget Testing Essentials

Always Set Screen Size

testWidgets('Test description', (tester) async { tester.view.physicalSize = const Size(1000, 1000); tester.view.devicePixelRatio = 1.0;

// Your test code });

Always Use Keys for Widget Finding

In source code:

ElevatedButton( key: const Key('saveButton'), onPressed: () {}, child: const Text('Save'), );

In test:

await tester.tap(find.byKey(const Key('saveButton'))); await tester.pumpAndSettle();

If a key doesn't exist in the source widget, add it before writing the test.

Loading → Content Transitions

when(mockService.fetchData()).thenAnswer((_) async => data);

await tester.pumpWidget(createTestWidget()); expect(find.byType(CircularProgressIndicator), findsOneWidget);

await tester.pumpAndSettle(); expect(find.byType(DataWidget), findsOneWidget);

Platform-Specific Testing

testWidgets('iOS specific behavior', (tester) async { debugDefaultTargetPlatformOverride = TargetPlatform.iOS;

await tester.pumpWidget(createTestWidget());

expect(find.byType(CupertinoButton), findsOneWidget);

debugDefaultTargetPlatformOverride = null; });

Riverpod Testing

Create Container with Overrides

final container = createContainer(overrides: [ repoProvider.overrideWith((ref) => mockRepo), ]);

Use the createContainer() helper from test/riverpod_container.dart which auto-disposes on tearDown.

Test Provider State

test('Given valid data, When state updates, Then reflects new value', () async { final notifier = container.read(provider.notifier);

notifier.updateState(newValue);

expect(container.read(provider).value!.property, newValue); });

Test Initial State

test('Given empty data, When building initial state, Then returns default state', () async { when(mockService.fetchData()).thenAnswer((_) async => []);

final container = createContainer(); final state = await container.read(provider.notifier).future;

expect(state.data, isEmpty); expect(state.isLoading, false); });

Stubbing Patterns

Success Scenarios

when(mockRepo.fetchFromDb()).thenAnswer(() async => mockData); when(mockApi.updateData(any, any, any)).thenAnswer(() async => true);

Failure Scenarios

when(mockRepo.fetchFromDb()).thenThrow(Exception('DB error')); when(mockApi.updateData(any, any, any)).thenAnswer((_) async => false);

Using Completers for Async Control

final completer = Completer<RegistrationModel>();

when(mockRepo.fetchData(any, any)).thenAnswer((_) => completer.future);

await tester.tap(find.text('Save')); await tester.pump();

expect(find.byType(CircularProgressIndicator), findsOneWidget);

completer.complete(const RegistrationModel(status: 'success')); await tester.pump();

expect(find.byType(CircularProgressIndicator), findsNothing);

Fakes vs Mocks

When to Use Fakes

Use fake implementations for consistent dummy behavior:

class FakeLogger extends ILogger { @override void writeInfoLog(String className, String method, String message) {}

@override void writeErrorLog(String className, String method, dynamic error, StackTrace? stack, [String? msg]) {} }

Register in test setup:

GetIt.I.registerSingleton<ILogger>(FakeLogger.i);

When to Use Mocks

Use mocks when you need to verify method calls or setup specific behaviors:

when(mockLogger.writeErrorLog(any, any, any, any)).thenReturn(null); verify(mockLogger.writeErrorLog('ClassName', 'methodName', exception, any)).called(1);

Database Testing with FakeDatabase

late MenuDAO menuDAO; late Database db; late IDatabase mockDatabase;

setUp(() async { await FakePathProviderPlatform.initialize(); PathProviderPlatform.instance = FakePathProviderPlatform();

mockDatabase = FakeDatabase(); db = await mockDatabase.database;

menuDAO = MenuDAO( dbManager: mockDatabase, logger: mockLogger, ); });

tearDown() async { await menuDAO.deleteTable(); if (GetIt.I.isRegistered<IDatabase>()) { await GetIt.I<IDatabase>().close(); } await GetIt.I.reset(); await FakePathProviderPlatform.cleanup(); }

Test Checklist

Before submitting tests, ensure:

Setup & Mocking:

  • Dependencies mocked (not providers)

  • SharedPreferences mocked if used

  • GetIt reset in tearDown

  • Streams closed in tearDown

  • Controllers disposed in tearDown

Widget Tests:

  • Keys added & used in widget tests

  • Screen size set (physicalSize + devicePixelRatio)

  • Platform overrides reset (debugDefaultTargetPlatformOverride = null)

  • Navigation tested if applicable

  • Dialogs/overlays tested if shown

Test Coverage:

  • Success & failure paths covered

  • Edge cases tested (null, empty, max values)

  • Loading states tested

  • Error states tested

  • Async handled correctly (await, Completer)

Code Quality:

  • Given-When-Then naming used

  • verify() or verifyNever() used where appropriate

  • No hardcoded delays (use pump/pumpAndSettle)

  • Tests are isolated (no dependencies between tests)

  • Tests are deterministic (same result every time)

Common Patterns

Verification Patterns

// Single call verify(mockService.method()).called(1);

// Multiple calls verify(mockService.method()).called(3);

// Never called verifyNever(mockService.method());

// Ordered calls verifyInOrder([ mockService.method1(), mockService.method2(), ]);

Testing Global State

import 'package:your_app/path/to/global_variables.dart' as global_variables;

setUp(() { global_variables.someGlobalVariable = initialValue; });

tearDown(() { global_variables.someGlobalVariable = initialValue; // Reset to default });

Testing Dispose/Cleanup

testWidgets('Given provider disposed, When container disposed, Then unsubscribes and cleans up', () async { final container = createContainer(); final notifier = container.read(provider.notifier); await notifier.future;

container.dispose();

verify(mockService.unsubscribe(any, any)).called(1); verify(mockService.dispose(any)).called(1); });

Running Tests

Run All Tests

flutter test --coverage

Or if using FVM:

fvm flutter test --coverage

Run Specific Test File

flutter test test/path/to/your_test.dart

Or if using FVM:

fvm flutter test test/path/to/your_test.dart

Run Specific Test by Name

flutter test --plain-name "Given valid data"

Or if using FVM:

fvm flutter test --plain-name "Given valid data"

Generate Coverage Report

flutter test --coverage genhtml coverage/lcov.info -o coverage/html

Or if using FVM:

fvm flutter test --coverage genhtml coverage/lcov.info -o coverage/html

Test Helpers and Utilities

Creating a Test Widget Wrapper

Widget createTestWidget(Widget child) { return MaterialApp( home: Scaffold( body: child, ), ); }

// With Riverpod Widget createTestWidgetWithProviders(Widget child, List<Override> overrides) { return ProviderScope( overrides: overrides, child: MaterialApp( home: Scaffold( body: child, ), ), ); }

Reusable Riverpod Container Helper

ProviderContainer createContainer({List<Override> overrides = const []}) { final container = ProviderContainer(overrides: overrides); addTearDown(container.dispose); return container; }

Finding Widgets by Type and Text

// Find by type expect(find.byType(CircularProgressIndicator), findsOneWidget);

// Find by text expect(find.text('Hello'), findsOneWidget);

// Find by key expect(find.byKey(const Key('myKey')), findsOneWidget);

// Find descendant expect( find.descendant( of: find.byType(Container), matching: find.text('Child'), ), findsOneWidget, );

// Find ancestor expect( find.ancestor( of: find.text('Child'), matching: find.byType(Container), ), findsOneWidget, );

Matchers for Better Assertions

// Common matchers expect(value, equals(expected)); expect(value, isNotNull); expect(value, isNull); expect(list, isEmpty); expect(list, isNotEmpty); expect(list, hasLength(3)); expect(list, contains('item')); expect(list, containsAll(['a', 'b'])); expect(value, greaterThan(5)); expect(value, lessThan(10)); expect(value, inRange(1, 10));

// Custom matchers expect(find.byType(Widget), findsOneWidget); expect(find.byType(Widget), findsNothing); expect(find.byType(Widget), findsWidgets); expect(find.byType(Widget), findsNWidgets(3));

Additional Testing Patterns

Testing with GetIt Dependency Injection

setUp(() { GetIt.I.registerSingleton<ApiService>(mockApiService); GetIt.I.registerLazySingleton<UserRepository>(() => UserRepository()); });

tearDown(() { GetIt.I.reset(); // Always reset GetIt between tests });

Testing Stream-Based Code

test('Given stream emits values, When listening, Then receives all values', () async { // Arrange final streamController = StreamController<int>(); final values = <int>[];

// Act streamController.stream.listen(values.add); streamController.add(1); streamController.add(2); streamController.add(3); await streamController.close();

// Wait for stream to complete await Future.delayed(Duration.zero);

// Assert expect(values, equals([1, 2, 3])); });

Testing Timer-Based Code

testWidgets('Given timer completes, When countdown finishes, Then shows message', (tester) async { await tester.pumpWidget(MyTimerWidget());

// Fast-forward time await tester.pump(const Duration(seconds: 5));

expect(find.text('Time is up!'), findsOneWidget); });

Testing Scrollable Widgets

testWidgets('Given long list, When scrolling, Then finds bottom item', (tester) async { await tester.pumpWidget(MyLongListWidget());

// Scroll until item is visible await tester.scrollUntilVisible( find.text('Item 99'), 500.0, );

expect(find.text('Item 99'), findsOneWidget); });

Resources

This skill includes reference files with detailed patterns and examples:

references/

  • layer_testing_patterns.md

  • Comprehensive examples for testing repositories, providers, DAOs, and services

  • widget_testing_guide.md

  • Detailed widget testing patterns with keys, screen size, and user interactions

  • riverpod_testing_guide.md

  • Advanced Riverpod provider testing patterns and state management testing

Refer to these references when you need specific implementation examples or encounter complex testing scenarios.

Source Transparency

This detail page is rendered from real SKILL.md content. Trust labels are metadata-based hints, not a safety guarantee.

Related Skills

Related by shared tags or category signals.

Coding

flutter-developer

No summary provided by upstream source.

Repository SourceNeeds Review
General

flutter-reviewer

No summary provided by upstream source.

Repository SourceNeeds Review
General

api-schema-zod

No summary provided by upstream source.

Repository SourceNeeds Review
General

lint-fix

No summary provided by upstream source.

Repository SourceNeeds Review