feat: WebSocket temps réel + Finance Workflow + corrections
- Task #6: WebSocket /ws/dashboard + Kafka events (5 topics) * Backend: KafkaEventProducer, KafkaEventConsumer * Mobile: WebSocketService (reconnection, heartbeat, typed events) * DashboardBloc: Auto-refresh depuis WebSocket events - Finance Workflow: approbations + budgets (backend + mobile) * Backend: entities, services, resources, migrations Flyway V6 * Mobile: features finance_workflow complète avec BLoC - Corrections DI: interfaces IRepository partout * IProfileRepository, IOrganizationRepository, IMembreRepository * GetIt configuré avec @injectable - Spec-Kit: constitution + templates mis à jour * .specify/memory/constitution.md enrichie * Templates agent, plan, spec, tasks, checklist - Nettoyage: fichiers temporaires supprimés Signed-off-by: lions dev Team
This commit is contained in:
@@ -0,0 +1,75 @@
|
||||
/// Tests unitaires pour DeleteAccount use case
|
||||
library delete_account_test;
|
||||
|
||||
import 'package:flutter_test/flutter_test.dart';
|
||||
import 'package:mockito/annotations.dart';
|
||||
import 'package:mockito/mockito.dart';
|
||||
import 'package:unionflow_mobile_apps/features/profile/domain/repositories/profile_repository.dart';
|
||||
import 'package:unionflow_mobile_apps/features/profile/domain/usecases/delete_account.dart';
|
||||
|
||||
@GenerateMocks([IProfileRepository])
|
||||
import 'delete_account_test.mocks.dart';
|
||||
|
||||
void main() {
|
||||
late DeleteAccount useCase;
|
||||
late MockIProfileRepository mockRepository;
|
||||
|
||||
setUp(() {
|
||||
mockRepository = MockIProfileRepository();
|
||||
useCase = DeleteAccount(mockRepository);
|
||||
});
|
||||
|
||||
group('DeleteAccount Use Case', () {
|
||||
const tMembreId = 'membre1';
|
||||
|
||||
test('should delete account successfully (soft delete)', () async {
|
||||
// Arrange
|
||||
when(mockRepository.deleteAccount(tMembreId))
|
||||
.thenAnswer((_) async => Future.value());
|
||||
|
||||
// Act
|
||||
await useCase(tMembreId);
|
||||
|
||||
// Assert
|
||||
verify(mockRepository.deleteAccount(tMembreId));
|
||||
verifyNoMoreInteractions(mockRepository);
|
||||
});
|
||||
|
||||
test('should throw exception when account not found', () async {
|
||||
// Arrange
|
||||
when(mockRepository.deleteAccount(any))
|
||||
.thenThrow(Exception('Compte non trouvé'));
|
||||
|
||||
// Act & Assert
|
||||
expect(
|
||||
() => useCase(tMembreId),
|
||||
throwsA(isA<Exception>()),
|
||||
);
|
||||
verify(mockRepository.deleteAccount(tMembreId));
|
||||
});
|
||||
|
||||
test('should throw exception when account is already deleted', () async {
|
||||
// Arrange
|
||||
when(mockRepository.deleteAccount(any))
|
||||
.thenThrow(Exception('Compte déjà désactivé'));
|
||||
|
||||
// Act & Assert
|
||||
expect(
|
||||
() => useCase(tMembreId),
|
||||
throwsA(isA<Exception>()),
|
||||
);
|
||||
});
|
||||
|
||||
test('should throw exception when deletion fails', () async {
|
||||
// Arrange
|
||||
when(mockRepository.deleteAccount(any))
|
||||
.thenThrow(Exception('Deletion failed'));
|
||||
|
||||
// Act & Assert
|
||||
expect(
|
||||
() => useCase(tMembreId),
|
||||
throwsException,
|
||||
);
|
||||
});
|
||||
});
|
||||
}
|
||||
@@ -0,0 +1,84 @@
|
||||
/// Tests unitaires pour GetProfile use case
|
||||
library get_profile_test;
|
||||
|
||||
import 'package:flutter_test/flutter_test.dart';
|
||||
import 'package:mockito/annotations.dart';
|
||||
import 'package:mockito/mockito.dart';
|
||||
import 'package:unionflow_mobile_apps/features/profile/domain/repositories/profile_repository.dart';
|
||||
import 'package:unionflow_mobile_apps/features/profile/domain/usecases/get_profile.dart';
|
||||
import 'package:unionflow_mobile_apps/features/members/data/models/membre_complete_model.dart';
|
||||
|
||||
@GenerateMocks([IProfileRepository])
|
||||
import 'get_profile_test.mocks.dart';
|
||||
|
||||
void main() {
|
||||
late GetProfile useCase;
|
||||
late MockIProfileRepository mockRepository;
|
||||
|
||||
setUp(() {
|
||||
mockRepository = MockIProfileRepository();
|
||||
useCase = GetProfile(mockRepository);
|
||||
});
|
||||
|
||||
group('GetProfile Use Case', () {
|
||||
final tMembre = MembreCompletModel(
|
||||
id: 'membre1',
|
||||
nom: 'Dupont',
|
||||
prenom: 'Jean',
|
||||
email: 'jean.dupont@example.com',
|
||||
telephone: '+33612345678',
|
||||
dateNaissance: DateTime(1990, 1, 1),
|
||||
);
|
||||
|
||||
test('should return current user profile from repository', () async {
|
||||
// Arrange
|
||||
when(mockRepository.getMe()).thenAnswer((_) async => tMembre);
|
||||
|
||||
// Act
|
||||
final result = await useCase();
|
||||
|
||||
// Assert
|
||||
expect(result, equals(tMembre));
|
||||
verify(mockRepository.getMe());
|
||||
verifyNoMoreInteractions(mockRepository);
|
||||
});
|
||||
|
||||
test('should return null when user is not authenticated', () async {
|
||||
// Arrange
|
||||
when(mockRepository.getMe()).thenAnswer((_) async => null);
|
||||
|
||||
// Act
|
||||
final result = await useCase();
|
||||
|
||||
// Assert
|
||||
expect(result, isNull);
|
||||
verify(mockRepository.getMe());
|
||||
});
|
||||
|
||||
test('should throw exception when repository throws', () async {
|
||||
// Arrange
|
||||
when(mockRepository.getMe()).thenThrow(Exception('Unauthorized'));
|
||||
|
||||
// Act & Assert
|
||||
expect(
|
||||
() => useCase(),
|
||||
throwsA(isA<Exception>()),
|
||||
);
|
||||
verify(mockRepository.getMe());
|
||||
});
|
||||
|
||||
test('should cache profile data on successful retrieval', () async {
|
||||
// Arrange
|
||||
when(mockRepository.getMe()).thenAnswer((_) async => tMembre);
|
||||
|
||||
// Act - Call twice
|
||||
final result1 = await useCase();
|
||||
final result2 = await useCase();
|
||||
|
||||
// Assert - Repository called twice (no caching at use case level)
|
||||
expect(result1, equals(tMembre));
|
||||
expect(result2, equals(tMembre));
|
||||
verify(mockRepository.getMe()).called(2);
|
||||
});
|
||||
});
|
||||
}
|
||||
@@ -0,0 +1,95 @@
|
||||
/// Tests unitaires pour UpdateAvatar use case
|
||||
library update_avatar_test;
|
||||
|
||||
import 'package:flutter_test/flutter_test.dart';
|
||||
import 'package:mockito/annotations.dart';
|
||||
import 'package:mockito/mockito.dart';
|
||||
import 'package:unionflow_mobile_apps/features/profile/domain/repositories/profile_repository.dart';
|
||||
import 'package:unionflow_mobile_apps/features/profile/domain/usecases/update_avatar.dart';
|
||||
import 'package:unionflow_mobile_apps/features/members/data/models/membre_complete_model.dart';
|
||||
|
||||
@GenerateMocks([IProfileRepository])
|
||||
import 'update_avatar_test.mocks.dart';
|
||||
|
||||
void main() {
|
||||
late UpdateAvatar useCase;
|
||||
late MockIProfileRepository mockRepository;
|
||||
|
||||
setUp(() {
|
||||
mockRepository = MockIProfileRepository();
|
||||
useCase = UpdateAvatar(mockRepository);
|
||||
});
|
||||
|
||||
group('UpdateAvatar Use Case', () {
|
||||
const tMembreId = 'membre1';
|
||||
const tPhotoUrl = 'https://example.com/avatar.jpg';
|
||||
|
||||
final tUpdatedMembre = MembreCompletModel(
|
||||
id: tMembreId,
|
||||
nom: 'Dupont',
|
||||
prenom: 'Jean',
|
||||
email: 'jean.dupont@example.com',
|
||||
photo: tPhotoUrl,
|
||||
);
|
||||
|
||||
test('should update avatar successfully', () async {
|
||||
// Arrange
|
||||
when(mockRepository.updateAvatar(tMembreId, tPhotoUrl))
|
||||
.thenAnswer((_) async => tUpdatedMembre);
|
||||
|
||||
// Act
|
||||
final result = await useCase(tMembreId, tPhotoUrl);
|
||||
|
||||
// Assert
|
||||
expect(result, equals(tUpdatedMembre));
|
||||
expect(result.photo, equals(tPhotoUrl));
|
||||
verify(mockRepository.updateAvatar(tMembreId, tPhotoUrl));
|
||||
verifyNoMoreInteractions(mockRepository);
|
||||
});
|
||||
|
||||
test('should handle empty photo URL', () async {
|
||||
// Arrange
|
||||
const emptyUrl = '';
|
||||
final emptyPhotoMembre = MembreCompletModel(
|
||||
id: tMembreId,
|
||||
nom: 'Dupont',
|
||||
prenom: 'Jean',
|
||||
email: 'jean.dupont@example.com',
|
||||
photo: emptyUrl,
|
||||
);
|
||||
when(mockRepository.updateAvatar(tMembreId, emptyUrl))
|
||||
.thenAnswer((_) async => emptyPhotoMembre);
|
||||
|
||||
// Act
|
||||
final result = await useCase(tMembreId, emptyUrl);
|
||||
|
||||
// Assert
|
||||
expect(result.photo, equals(emptyUrl));
|
||||
verify(mockRepository.updateAvatar(tMembreId, emptyUrl));
|
||||
});
|
||||
|
||||
test('should throw exception when member not found', () async {
|
||||
// Arrange
|
||||
when(mockRepository.updateAvatar(any, any))
|
||||
.thenThrow(Exception('Membre non trouvé'));
|
||||
|
||||
// Act & Assert
|
||||
expect(
|
||||
() => useCase(tMembreId, tPhotoUrl),
|
||||
throwsA(isA<Exception>()),
|
||||
);
|
||||
});
|
||||
|
||||
test('should throw exception when upload fails', () async {
|
||||
// Arrange
|
||||
when(mockRepository.updateAvatar(any, any))
|
||||
.thenThrow(Exception('Upload failed'));
|
||||
|
||||
// Act & Assert
|
||||
expect(
|
||||
() => useCase(tMembreId, tPhotoUrl),
|
||||
throwsException,
|
||||
);
|
||||
});
|
||||
});
|
||||
}
|
||||
@@ -0,0 +1,93 @@
|
||||
/// Tests unitaires pour UpdatePreferences use case
|
||||
library update_preferences_test;
|
||||
|
||||
import 'package:flutter_test/flutter_test.dart';
|
||||
import 'package:mockito/annotations.dart';
|
||||
import 'package:mockito/mockito.dart';
|
||||
import 'package:unionflow_mobile_apps/features/profile/domain/repositories/profile_repository.dart';
|
||||
import 'package:unionflow_mobile_apps/features/profile/domain/usecases/update_preferences.dart';
|
||||
|
||||
@GenerateMocks([IProfileRepository])
|
||||
import 'update_preferences_test.mocks.dart';
|
||||
|
||||
void main() {
|
||||
late UpdatePreferences useCase;
|
||||
late MockIProfileRepository mockRepository;
|
||||
|
||||
setUp(() {
|
||||
mockRepository = MockIProfileRepository();
|
||||
useCase = UpdatePreferences(mockRepository);
|
||||
});
|
||||
|
||||
group('UpdatePreferences Use Case', () {
|
||||
const tMembreId = 'membre1';
|
||||
final tPreferences = {
|
||||
'language': 'fr',
|
||||
'theme': 'dark',
|
||||
'notifications': true,
|
||||
'emailNotifications': false,
|
||||
};
|
||||
|
||||
final tUpdatedPreferences = {
|
||||
...tPreferences,
|
||||
'lastUpdated': '2026-03-14T10:00:00Z',
|
||||
};
|
||||
|
||||
test('should update preferences successfully', () async {
|
||||
// Arrange
|
||||
when(mockRepository.updatePreferences(tMembreId, tPreferences))
|
||||
.thenAnswer((_) async => tUpdatedPreferences);
|
||||
|
||||
// Act
|
||||
final result = await useCase(tMembreId, tPreferences);
|
||||
|
||||
// Assert
|
||||
expect(result, equals(tUpdatedPreferences));
|
||||
expect(result['language'], equals('fr'));
|
||||
expect(result['theme'], equals('dark'));
|
||||
verify(mockRepository.updatePreferences(tMembreId, tPreferences));
|
||||
verifyNoMoreInteractions(mockRepository);
|
||||
});
|
||||
|
||||
test('should update partial preferences', () async {
|
||||
// Arrange
|
||||
final partialPrefs = {'theme': 'light'};
|
||||
final expectedResult = {'theme': 'light'};
|
||||
when(mockRepository.updatePreferences(tMembreId, partialPrefs))
|
||||
.thenAnswer((_) async => expectedResult);
|
||||
|
||||
// Act
|
||||
final result = await useCase(tMembreId, partialPrefs);
|
||||
|
||||
// Assert
|
||||
expect(result['theme'], equals('light'));
|
||||
verify(mockRepository.updatePreferences(tMembreId, partialPrefs));
|
||||
});
|
||||
|
||||
test('should handle empty preferences map', () async {
|
||||
// Arrange
|
||||
final emptyPrefs = <String, dynamic>{};
|
||||
when(mockRepository.updatePreferences(tMembreId, emptyPrefs))
|
||||
.thenAnswer((_) async => emptyPrefs);
|
||||
|
||||
// Act
|
||||
final result = await useCase(tMembreId, emptyPrefs);
|
||||
|
||||
// Assert
|
||||
expect(result, isEmpty);
|
||||
verify(mockRepository.updatePreferences(tMembreId, emptyPrefs));
|
||||
});
|
||||
|
||||
test('should throw exception when update fails', () async {
|
||||
// Arrange
|
||||
when(mockRepository.updatePreferences(any, any))
|
||||
.thenThrow(Exception('Failed to update preferences'));
|
||||
|
||||
// Act & Assert
|
||||
expect(
|
||||
() => useCase(tMembreId, tPreferences),
|
||||
throwsException,
|
||||
);
|
||||
});
|
||||
});
|
||||
}
|
||||
@@ -0,0 +1,97 @@
|
||||
/// Tests unitaires pour UpdateProfile use case
|
||||
library update_profile_test;
|
||||
|
||||
import 'package:flutter_test/flutter_test.dart';
|
||||
import 'package:mockito/annotations.dart';
|
||||
import 'package:mockito/mockito.dart';
|
||||
import 'package:unionflow_mobile_apps/features/profile/domain/repositories/profile_repository.dart';
|
||||
import 'package:unionflow_mobile_apps/features/profile/domain/usecases/update_profile.dart';
|
||||
import 'package:unionflow_mobile_apps/features/members/data/models/membre_complete_model.dart';
|
||||
|
||||
@GenerateMocks([IProfileRepository])
|
||||
import 'update_profile_test.mocks.dart';
|
||||
|
||||
void main() {
|
||||
late UpdateProfile useCase;
|
||||
late MockIProfileRepository mockRepository;
|
||||
|
||||
setUp(() {
|
||||
mockRepository = MockIProfileRepository();
|
||||
useCase = UpdateProfile(mockRepository);
|
||||
});
|
||||
|
||||
group('UpdateProfile Use Case', () {
|
||||
const tMembreId = 'membre1';
|
||||
final tMembre = MembreCompletModel(
|
||||
id: tMembreId,
|
||||
nom: 'Dupont',
|
||||
prenom: 'Jean',
|
||||
email: 'jean.dupont@example.com',
|
||||
telephone: '+33612345678',
|
||||
dateNaissance: DateTime(1990, 1, 1),
|
||||
);
|
||||
|
||||
final tUpdatedMembre = MembreCompletModel(
|
||||
id: tMembreId,
|
||||
nom: 'Dupont',
|
||||
prenom: 'Jean',
|
||||
email: 'jean.dupont@example.com',
|
||||
telephone: '+33698765432', // Updated phone
|
||||
dateNaissance: DateTime(1990, 1, 1),
|
||||
adresse: '123 Rue de Paris', // Added address
|
||||
);
|
||||
|
||||
test('should update profile successfully', () async {
|
||||
// Arrange
|
||||
when(mockRepository.updateProfile(tMembreId, tMembre))
|
||||
.thenAnswer((_) async => tUpdatedMembre);
|
||||
|
||||
// Act
|
||||
final result = await useCase(tMembreId, tMembre);
|
||||
|
||||
// Assert
|
||||
expect(result, equals(tUpdatedMembre));
|
||||
verify(mockRepository.updateProfile(tMembreId, tMembre));
|
||||
verifyNoMoreInteractions(mockRepository);
|
||||
});
|
||||
|
||||
test('should update only specified fields', () async {
|
||||
// Arrange
|
||||
when(mockRepository.updateProfile(any, any))
|
||||
.thenAnswer((_) async => tUpdatedMembre);
|
||||
|
||||
// Act
|
||||
final result = await useCase(tMembreId, tMembre);
|
||||
|
||||
// Assert
|
||||
expect(result.telephone, equals('+33698765432'));
|
||||
expect(result.adresse, equals('123 Rue de Paris'));
|
||||
verify(mockRepository.updateProfile(tMembreId, tMembre));
|
||||
});
|
||||
|
||||
test('should throw exception when profile not found', () async {
|
||||
// Arrange
|
||||
when(mockRepository.updateProfile(any, any))
|
||||
.thenThrow(Exception('Profil non trouvé'));
|
||||
|
||||
// Act & Assert
|
||||
expect(
|
||||
() => useCase(tMembreId, tMembre),
|
||||
throwsA(isA<Exception>()),
|
||||
);
|
||||
verify(mockRepository.updateProfile(tMembreId, tMembre));
|
||||
});
|
||||
|
||||
test('should throw exception when update fails', () async {
|
||||
// Arrange
|
||||
when(mockRepository.updateProfile(any, any))
|
||||
.thenThrow(Exception('Network error'));
|
||||
|
||||
// Act & Assert
|
||||
expect(
|
||||
() => useCase(tMembreId, tMembre),
|
||||
throwsException,
|
||||
);
|
||||
});
|
||||
});
|
||||
}
|
||||
Reference in New Issue
Block a user