Contents
- Structuring Code for Testability
- Managing Dev Dependencies
- Generating Mock Files
- Stubbing and Verification Best Practices
- Workflow: Generating Mocks and Validating Tests
- Examples
Structuring Code for Testability
To write effective unit tests, structure your codebase using dependency injection. Isolate operations that interact with physical layers (like disk storage, external servers, and platform channels) so they can be replaced by mock objects at test time:
- Constructor Injection: Pass all service dependencies (e.g. HTTP clients, database helper clients) into class constructors rather than instantiating them directly within the class.
- Interface Segregation: Define clear, abstract base classes representing your service contracts. Mocks should ideally target these abstract contracts rather than concrete service implementations.
Managing Dev Dependencies
Configure your pubspec.yaml to specify the packages required for code generation and mock testing:
- Add target testing frameworks and generators under
dev_dependencies:dart pub add dev:test dev:mockito dev:build_runner - Import Mockito annotations inside your test files:
import 'package:mockito/annotations.dart'; import 'package:mockito/mockito.dart';
Generating Mock Files
Leverage package:mockito alongside build_runner to generate mock structures automatically:
- GenerateNiceMocks: Always use the
@GenerateNiceMocksannotation instead of the legacy@GenerateMocks. Nice mocks automatically return null or matching default values instead of throwing “MissingStubException” when a method is invoked without a pre-configured stub. - MockSpec Configuration: Annotate your test file’s entry point with
@GenerateNiceMocks([MockSpec<YourService>()]). - Mirror Extension Import: Import the matching generated file using the
.mocks.dartsuffix (e.g.import 'service_test.mocks.dart'). - Run Generator: Trigger code generation via the CLI:
dart run build_runner build --delete-conflicting-outputs
Stubbing and Verification Best Practices
Write robust mock interactions by adhering to these guidelines:
- Future and Stream Stubbing: When stubbing methods that return a
Futureor aStream, always use.thenAnswer((_) async => value). Never use.thenReturn()for asynchronous return values, as this causes runtime cast errors. - Invocation Tracking: Use the
verify()API to verify that specific methods were invoked. Call.called(number)to assert precise invocation counts. - Unused Assertions: Use
verifyNever()orverifyNoMoreInteractions(mock)to guarantee that no unexpected actions occurred during the test lifecycle.
Workflow: Generating Mocks and Validating Tests
Follow this checklist to establish mock configurations:
- [ ] Constructor check: Ensure the target service class accepts dependencies via its constructor.
- [ ] Write test file: Create
test/feature_test.dartand import Mockito along with testing libraries. - [ ] Annotate entrypoint: Add
@GenerateNiceMocks([MockSpec<TargetService>()])abovemain(). - [ ] Declare mock import: Add the import statement targeting
feature_test.mocks.dart. - [ ] Generate code: Run
dart run build_runner buildin your terminal to create the mocked file. - [ ] Instantiate mocks: In the
setUpcallback, instantiate the generated mock class (e.g.mockService = MockTargetService()). - [ ] Apply stubbing: Configure behavior in your test cases using
when(). - [ ] Execute and assert: Call the system under test, verifying outputs via
expect(). - [ ] Verify calls: Use
verify()to assert mock interactions.
Examples
Complete Mocked Test Suit (Mockito)
This example shows how to mock a remote API client to test a database synchronization service.
import 'package:test/test.dart';
import 'package:mockito/annotations.dart';
import 'package:mockito/mockito.dart';
import 'package:http/http.dart' as http;
// Define an abstract contract for a user repository
abstract class UserRepository {
Future<String> getUserName(int id);
}
// System Under Test consuming the repository
class UserService {
final UserRepository repository;
UserService(this.repository);
Future<String> fetchDisplayName(int id) async {
try {
final name = await repository.getUserName(id);
return 'User: $name';
} catch (_) {
return 'Unknown User';
}
}
}
// 1. Annotate to generate MockUserRepository
@GenerateNiceMocks([MockSpec<UserRepository>()])
import 'user_service_test.mocks.dart';
void main() {
group('UserService', () {
late MockUserRepository mockRepo;
late UserService userService;
setUp(() {
mockRepo = MockUserRepository();
userService = UserService(mockRepo);
});
test('returns formatted display name when repository succeeds', () async {
// 2. Arrange: Stub the async getUserName method
when(mockRepo.getUserName(42)).thenAnswer(
(_) async => 'Alice',
);
// 3. Act: Run target method
final displayName = await userService.fetchDisplayName(42);
// 4. Assert: Validate outcomes
expect(displayName, equals('User: Alice'));
// 5. Verify: Assert the repository method was invoked with correct arguments
verify(mockRepo.getUserName(42)).called(1);
});
test('returns fallback string when repository throws exception', () async {
// Arrange: Stub to throw error
when(mockRepo.getUserName(any)).thenThrow(
Exception('Connection Timeout'),
);
// Act
final displayName = await userService.fetchDisplayName(99);
// Assert
expect(displayName, equals('Unknown User'));
});
});
}